遍历时修改集合会抛ConcurrentModificationException,应使用Iterator.remove或removeIf;2. 可变对象作HashMap的key会导致查找失败,应使用不可变对象或确保hashCode和equals稳定;3. 忽视null处理可能引发空指针,应避免插入null或使用Optional;4. 未预设初始容量导致频繁扩容,应根据数据量初始化ArrayList或HashMap;5. 多线程下非同步访问集合不安全,应使用并发集合类如ConcurrentHashMap。

Java集合操作在日常开发中非常频繁,但稍不注意就会引发各种问题。这些问题轻则导致程序运行缓慢,重则引发崩溃或数据错误。以下是一些常见的错误及其避免方法,帮助你写出更健壮的代码。
1. 在遍历过程中修改集合
这是最常见的错误之一。直接在 for-each 循环中调用集合的 add 或 remove 方法会抛出 ConcurrentModificationException。
示例:使用普通 for-each 遍历时删除元素:
for (String s : list) { if ("bad".equals(s)) list.remove(s); }
立即学习“Java免费学习笔记(深入)”;
这会触发异常,因为迭代器检测到了结构上的修改。
避免方法:
- 使用 Iterator 的 remove 方法:它设计用于安全删除。
- 先收集要删除的元素,遍历结束后统一处理。
- 使用 removeIf 方法(Java 8+),简洁且安全。
推荐写法:
list.removeIf("bad"::equals);
2. 使用可变对象作为 HashMap 的 key
当把一个对象作为 HashMap 的 key,并在之后修改其内容(尤其是影响 hashCode 或 equals 的字段),会导致该 key 无法被正确查找,甚至造成内存泄漏。
避免方法:
- 尽量使用不可变对象(如 String、Integer)作为 key。
- 若必须使用自定义对象,确保其关键字段是 final 的,并正确重写 hashCode 和 equals。
- 避免在放入 Map 后修改 key 对象的状态。
3. 忽视 null 值的处理
某些集合实现不允许 null 值,比如 ConcurrentHashMap,向其中 put null 会抛出 NullPointerException。而 ArrayList 虽然允许 null,但在后续操作中容易引发空指针异常。
避免方法:
- 明确业务逻辑是否需要支持 null,避免随意插入。
- 使用 Objects.requireNonNull 检查输入参数。
- 优先考虑使用 Optional 或默认值代替 null。
4. 错误地初始化集合大小
默认情况下,ArrayList 和 HashMap 都有初始容量(通常是 10 和 16)。当元素数量远超初始值时,会频繁触发扩容,带来性能损耗。
避免方法:
- 预估数据量,在创建时指定初始容量。
- 例如:new ArrayList(1000) 或 new HashMap(512)。
- 对于 HashMap,还可设置合适的加载因子以平衡空间与性能。
5. 多线程环境下未同步访问集合
ArrayList、HashMap 等是非线程安全的。多线程并发修改时可能出现数据错乱、死循环甚至 JVM 崩溃(如 HashMap 扩容时的链表成环问题)。
避免方法:
- 使用 Collections.synchronizedList 或 synchronizedMap 包装。
- 优先选择并发集合类,如 CopyOnWriteArrayList、ConcurrentHashMap。
- 根据场景选择合适的数据结构,避免过度同步影响性能。
基本上就这些。掌握这些常见陷阱并养成良好习惯,能显著提升代码的稳定性和可维护性。细节决定成败,尤其是在集合操作这种高频场景中。










