List有序可重复,Set无序去重,Map存储键值对;遍历时需用Iterator删除元素;HashMap依赖hashCode和equals,自定义键需重写;根据场景选择集合类。

Java集合框架是每个初学者在学习过程中必须掌握的核心内容之一。但由于其种类繁多、接口与实现类关系复杂,初学者常常会遇到一些典型问题。以下是几个常见困惑及其解答,帮助你更好地理解和使用Java集合。
1. List、Set和Map有什么区别?
这是最常被问到的问题。理解三者之间的差异是使用集合的基础。
- List 是有序集合,允许重复元素,可以通过索引访问元素。常用实现类有 ArrayList 和 LinkedList。
- Set强> 是无序集合,不允许重复元素。常用实现类有 HashSet(基于哈希表)和 TreeSet(有序,基于红黑树)。
- Map 不是 Collection 的子接口,它存储的是键值对(key-value),键不能重复。常用实现类有 HashMap 和 TreeMap。
简单记:List像数组可重复,Set去重不保证顺序,Map用键找值。
2. 为什么遍历集合时不能随便删除元素?
很多初学者在使用 for-each 循环遍历 List 或 Set 时,尝试直接调用 remove() 方法删除元素,结果抛出 ConcurrentModificationException。
立即学习“Java免费学习笔记(深入)”;
这是因为普通集合类(如 ArrayList、HashSet)不是线程安全的,且迭代器会检测结构性修改。
正确做法:
- 使用 Iterator 的 remove() 方法。
- 改用 ListIterator(适用于List,支持双向遍历和修改)。
- 先收集要删除的元素,遍历结束后再删除。
- 使用支持并发修改的集合,如 CopyOnWriteArrayList(适合读多写少场景)。
3. HashMap的工作原理是什么?为什么键要重写hashCode和equals?
HashMap 内部基于哈希表实现,通过 key 的 hashCode() 计算存储位置,用 equals() 判断键是否相等。
如果你自定义对象作为键但没有重写这两个方法,将使用 Object 类的默认实现,可能导致:
- 不同对象即使“逻辑相等”也被视为不同键(因为内存地址不同)。
- 无法正确找到已存入的值。
所以,作为 HashMap 的键,必须同时重写 hashCode() 和 equals() 方法,并保持一致性:如果两个对象 equals 返回 true,它们的 hashCode 必须相同。
4. 如何选择合适的集合类?
面对多种实现类,初学者容易不知所措。选择应根据使用场景判断:
- 需要按索引访问?选 ArrayList。
- 频繁在中间插入或删除?考虑 LinkedList(但实际中 ArrayList 通常更快)。
- 要去重?用 HashSet;还要排序?用 TreeSet。
- 存键值对?优先 HashMap;需要按键排序?用 TreeMap。
- 多线程环境?考虑 ConcurrentHashMap 而不是 Hashtable。
大多数情况下,ArrayList、HashSet、HashMap 就够用了。
基本上就这些。刚接触集合时觉得杂乱很正常,关键是动手写代码,结合调试观察行为。随着使用增多,你会自然记住哪些该用、哪些要避免。










