选集合类需据场景定:存什么、怎么用、是否线程安全。顺序存取查下标用ArrayList;去重查存在用HashSet(快)或TreeSet(排序);需去重且保插入序用LinkedHashSet;多线程优先ConcurrentHashMap或CopyOnWriteArrayList。

选集合类,关键看三点:存什么、怎么用、是否线程安全。别一上来就用 ArrayList 或 HashMap,先想清楚场景。
要按顺序存取,且频繁查下标 → 用 ArrayList
ArrayList 底层是数组,支持 O(1) 随机访问,适合“读多写少”、按索引操作的场景。但尾部以外的增删是 O(n),因为要搬数据。
- 适合:日志列表、配置项列表、结果集缓存(不常改)
- 注意:初始容量设合理(比如知道大概存 500 条,就
new ArrayList(500)),避免反复扩容 - 别用它做队列(比如反复
remove(0)),性能会掉得厉害
要保证唯一、快速查存在性 → 优先考虑 HashSet 或 TreeSet
HashSet 基于哈希表,增删查平均 O(1),但不保证顺序;TreeSet 基于红黑树,自动排序,增删查 O(log n)。
- 去重计数、权限校验、ID 缓存 → 用
HashSet - 需要自然序或自定义排序(如排行榜、时间范围查询)→ 用
TreeSet - 注意:
HashSet要求元素正确实现hashCode()和equals()
要按插入顺序遍历,又想要去重 → LinkedHashSet 是答案
它保留了 HashSet 的查找效率,同时用链表维护插入顺序,迭代时顺序稳定。
立即学习“Java免费学习笔记(深入)”;
- 适合:最近访问记录、请求路径去重+保序、配置项加载顺序敏感的场景
- 内存略高于
HashSet(多一个指针链),但多数业务中这点开销可忽略 - 不是为了“看起来有序”而用,而是真有顺序语义需求时才选
多线程环境下别乱用非线程安全集合
ArrayList、HashMap、HashSet 等默认都不安全。高并发写入可能丢数据或抛 ConcurrentModificationException。
- 读多写少 + 要求强一致性 → 用
CopyOnWriteArrayList(适合监听器列表、配置快照) - 高频读写 + 接受弱一致性 →
ConcurrentHashMap(推荐,JDK8 后性能好、分段锁升级为 CAS + synchronized) - 简单计数或状态标记 →
AtomicInteger、ConcurrentLinkedQueue比包装集合更轻量 - 别用
Collections.synchronizedXxx()包一层就以为万事大吉——它只保证单个方法原子,复合操作(如“检查再插入”)仍需手动同步
基本上就这些。集合不是越“高级”越好,而是越贴合场景越稳。写代码前花半分钟想想:我到底要啥?查得快?排好序?能并发?还是只要简单装一装?答案清楚了,选型自然就出来了。










