
在软件开发中,我们经常会遇到需要根据特定条件从一个集合(如 ArrayList)中查找并移除元素,同时执行其他相关操作的场景。一个常见的挑战是在迭代过程中动态修改列表,这可能导致索引错乱、元素跳过或 ConcurrentModificationException 等问题。本教程将通过一个车辆停放的实际案例,演示一种健壮且有效的解决方案,以确保所有元素都被正确处理并最终清空列表。
假设我们有一个 vehicles 列表,其中包含了所有待停放的车辆对象。同时,我们还有一个 garage 列表,代表了所有可用的车库空间。我们的目标是:
为了实现上述目标,我们需要一个多层循环结构:
以下是实现上述策略的Java代码示例:
import java.util.ArrayList;
import java.util.List;
// 假设 Vehicle 和 Garage 类已经定义,并包含相应的方法
class Vehicle {
private int space;
private String vehicleType;
private String destinationSpace; // 假设有目的地空间属性
public Vehicle(int space, String vehicleType, String destinationSpace) {
this.space = space;
this.vehicleType = vehicleType;
this.destinationSpace = destinationSpace;
}
public int getSpace() {
return space;
}
public String getvehiclesType() {
return vehicleType;
}
public String getDestinationSpace() {
return destinationSpace;
}
@Override
public String toString() {
return "Vehicle [type=" + vehicleType + ", space=" + space + "]";
}
}
class Garage {
private int space;
private String garageType; // 假设车库有类型限制
private int limit; // 车库容量
private List<Vehicle> parkedVehicles; // 停放的车辆
public Garage(int space, String garageType, int limit) {
this.space = space;
this.garageType = garageType;
this.limit = limit;
this.parkedVehicles = new ArrayList<>();
}
public int getSpace() {
return space;
}
public boolean garageRequest(String vehicleType) {
// 假设车库类型与车辆类型匹配
return this.garageType.equals(vehicleType);
}
public int getLimit() {
return limit - parkedVehicles.size(); // 返回剩余容量
}
public void addvehicles(Vehicle v) {
if (getLimit() > 0) {
parkedVehicles.add(v);
System.out.println("Parked " + v + " in Garage [space=" + space + ", type=" + garageType + "]");
}
}
public List<Vehicle> getCarry() {
return parkedVehicles;
}
// 假设有方法来设置当前空间或移除车辆,但在此解决方案中未直接使用
public void setCurrentSpace(String newSpace) {
// 示例方法,实际逻辑根据需求实现
System.out.println("Garage " + space + " changed current space to " + newSpace);
}
public boolean removeVehicles(Vehicle v) {
boolean removed = parkedVehicles.remove(v);
if (removed) {
System.out.println("Removed " + v + " from Garage [space=" + space + ", type=" + garageType + "]");
}
return removed;
}
@Override
public String toString() {
return "Garage [space=" + space + ", type=" + garageType + ", available=" + getLimit() + "]";
}
}
public class VehicleParkingManager {
private List<Vehicle> vehicles;
private List<Garage> garage;
public VehicleParkingManager(List<Vehicle> vehicles, List<Garage> garage) {
this.vehicles = vehicles;
this.garage = garage;
}
public void parkAllVehicles() {
System.out.println("--- Starting Parking Process ---");
System.out.println("Initial Vehicles: " + vehicles.size());
System.out.println("Available Garages: " + garage.size());
while (vehicles.size() > 0) { // 外层循环:只要还有车辆未停放,就继续尝试
boolean parkedInThisIteration = false; // 标记本轮是否有车辆被停放
for (int i = 0; i < vehicles.size(); i++) { // 内层循环:遍历当前待停放车辆列表
Vehicle v = vehicles.get(i); // 获取当前车辆
boolean vehicleParked = false; // 标记当前车辆是否已停放
for (int j = 0; j < garage.size(); j++) { // 最内层循环:遍历所有车库
Garage g = garage.get(j); // 获取当前车库
// 检查停放条件:车库空间匹配、车辆类型匹配、车库有容量
if (g.getSpace() == v.getSpace() && g.garageRequest(v.getvehiclesType()) && g.getLimit() > 0) {
g.addvehicles(v); // 将车辆停放到车库
vehicles.remove(i); // 从待停放列表中移除已停放的车辆
i--; // 关键:移除元素后,当前索引i指向了下一个元素,为了不跳过,需要将i减1
parkedInThisIteration = true;
vehicleParked = true;
break; // 跳出车库循环,因为当前车辆已停放
}
}
// 如果当前车辆在本轮循环中没有找到车库停放,则继续检查下一辆车
// 如果车辆已被停放,且i被调整,for循环的i++会抵消调整,继续正确迭代
}
if (!parkedInThisIteration && vehicles.size() > 0) {
System.out.println("No more vehicles could be parked in this iteration. Remaining vehicles: " + vehicles.size());
// 如果一整轮for循环都没有车辆被停放,且列表仍不为空,说明无法再停放,跳出while循环
break;
}
}
System.out.println("--- Parking Process Finished ---");
System.out.println("Remaining Vehicles: " + vehicles.size());
}
public static void main(String[] args) {
List<Vehicle> vehicles = new ArrayList<>();
vehicles.add(new Vehicle(1, "Car", "A1"));
vehicles.add(new Vehicle(2, "Truck", "B2"));
vehicles.add(new Vehicle(1, "Car", "A3"));
vehicles.add(new Vehicle(3, "Motorcycle", "C1"));
vehicles.add(new Vehicle(2, "Car", "B4")); // 无法停放,因为车库2只接受Truck
vehicles.add(new Vehicle(1, "Car", "A5")); // 无法停放,因为车库1容量已满
List<Garage> garages = new ArrayList<>();
garages.add(new Garage(1, "Car", 1)); // 只能停一辆Car
garages.add(new Garage(2, "Truck", 2)); // 只能停两辆Truck
garages.add(new Garage(3, "Motorcycle", 1)); // 只能停一辆Motorcycle
VehicleParkingManager manager = new VehicleParkingManager(vehicles, garages);
manager.parkAllVehicles();
System.out.println("\n--- Final State ---");
System.out.println("Vehicles left in list: " + manager.vehicles);
for (Garage g : garages) {
System.out.println(g + " Parked vehicles: " + g.getCarry());
}
}
}注意: 在原始提供的答案中,vehicles.remove(i) 后没有 i--。如果 for 循环是 for (int i = 0; i < vehicles.size(); i++) 且发生 remove(i),那么列表中的后续元素会向前移动,导致下一个 i++ 跳过一个元素。然而,原始答案的 while (vehicles.size() > 0) 循环会确保 for 循环在列表未清空时会重新从 i=0 开始,从而最终处理所有元素。为了更直接且避免潜在的跳过,我在示例代码中加入了 i--,这是在正向遍历并删除时常用的技巧。
while (vehicles.size() > 0):
for (int i = 0; i < vehicles.size(); i++):
**
以上就是精准匹配与移除:高效清空列表的车辆停放策略的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号