Java集合常见陷阱包括:遍历时直接删除元素引发ConcurrentModificationException,应使用迭代器remove()或removeIf();HashMap用可变对象作key导致查找失败,需保证hashCode/equals一致性且优先用不可变对象;ArrayList频繁add引发多次扩容,应预设初始容量;误用原始类型集合丢失泛型安全,须始终声明泛型类型。
Java集合用得不熟,很容易踩坑——不是空指针报错,就是遍历时修改出 ConcurrentModificationException,再或者用错类型导致性能暴跌。关键不在“会不会用”,而在“有没有意识到这些隐性陷阱”。
这是最经典也最容易复现的错误:用 for-each 或 Iterator 遍历时,调用集合自身的 remove() 方法(如 list.remove(obj)),会触发 ConcurrentModificationException。因为迭代器内部有 modCount 校验机制,而集合直接修改会改变结构但不通知迭代器。
如果把自定义对象(比如 Person)当 HashMap 的 key,但没重写 hashCode() 和 equals(),或者重写了却让 hash 值随对象字段变化(比如用含可变字段的计算逻辑),
就会出现 put 进去后 get 不回来的情况——因为 hash 槽变了,但对象还留在旧位置。
ArrayList 底层是数组,初始容量通常为 10。每次扩容要新建数组 + 复制元素,开销不小。如果事先知道大概数量(比如读取 5000 行文件),却不指定初始容量,可能触发 5~6 次扩容。
写成 List list = new ArrayList(); 看似能编译,但等于主动关闭了编译期类型检查。往里加 String、Integer、File 都不会报错,取出来却要强转,运行时 ClassCastException 风险陡增,IDE 也无法提示问题。
基本上就这些——不复杂,但容易忽略。避开它们,集合代码就稳了一大半。