在Java中如何选对合适的集合类_Java集合选择参考指南

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

选集合类,关键看三点:存什么、怎么用、是否线程安全。别一上来就用 ArrayListHashMap,先想清楚场景。

要按顺序存取,且频繁查下标 → 用 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 的查找效率,同时用链表维护插入顺序,迭代时顺序稳定。

  • 适合:最近访问记录、请求路径去重+保序、配置项加载顺序敏感的场景
  • 内存略高于 HashSet(多一个指针链),但多数业务中这点开销可忽略
  • 不是为了“看起来有序”而用,而是真有顺序语义需求时才选

多线程环境下别乱用非线程安全集合

ArrayListHashMapHashSet 等默认都不安全。高并发写入可能丢数据或抛 ConcurrentModificationException

  • 读多写少 + 要求强一致性 → 用 CopyOnWriteArrayList(适合监听器列表、配置快照)
  • 高频读写 + 接受弱一致性 → ConcurrentHashMap(推荐,JDK8 后性能好、分段锁升级为 CAS + synchronized)
  • 简单计数或状态标记 → AtomicIntegerConcurrentLinkedQueue 比包装集合更轻量
  • 别用 Collections.synchronizedXxx() 包一层就以为万事大吉——它只保证单个方法原子,复合操作(如“检查再插入”)仍需手动同步

基本上就这些。集合不是越“高级”越好,而是越贴合场景越稳。写代码前花半分钟想想:我到底要啥?查得快?排好序?能并发?还是只要简单装一装?答案清楚了,选型自然就出来了。