登录
首页 >  文章 >  java教程

Java同步集合与并发编程全解析

时间:2026-03-01 09:10:41 121浏览 收藏

本文深入剖析了Java并发编程中几类关键同步集合的底层原理与实战陷阱:ConcurrentHashMap凭借分段锁或CAS+synchronized机制实现高并发下的细粒度锁和无锁读,远胜于Hashtable的全局锁瓶颈;CopyOnWriteArrayList专为读多写少且需安全遍历的场景设计,但误用会导致性能灾难;Collections.synchronizedList仅保障单方法原子性,复合操作和迭代必须手动加锁;BlockingQueue则通过与线程池的协同,以有界/无界策略精细调控任务缓冲与资源水位——每一种选择都直指真实业务中的性能、一致性与稳定性权衡,是高并发系统避坑与提效的必修课。

Java集合框架中的同步集合与并发编程

ConcurrentHashMap 为什么比 Hashtable 更适合高并发场景

因为 ConcurrentHashMap 不是简单地给整个表加锁,而是采用分段锁(JDK 7)或更轻量的 CAS + synchronized(JDK 8+)策略,只锁定发生修改的桶(bucket),读操作完全无锁。而 Hashtable 所有方法都用 synchronized 修饰,意味着每次 get()put() 都要竞争同一把全局锁。

实操建议:

  • 如果项目用 JDK 8+,ConcurrentHashMap 是默认首选;它支持 computeIfAbsent()merge() 等原子操作,避免手动同步
  • 不要用 ConcurrentHashMapsize() 做精确计数判断——它返回的是估算值,多线程下可能不一致;需要精确大小时改用 mappingCount()
  • ConcurrentHashMap 不允许 null 作为 key 或 value,否则抛 NullPointerException;而 HashMap 允许一个 null key

CopyOnWriteArrayList 在什么场景下真正有用

它适用于「读多写少」且遍历中可能被修改的场景,比如事件监听器列表、配置白名单缓存。每次写操作(add()remove())都会复制整个底层数组,所以写开销大,但迭代器永不抛 ConcurrentModificationException,也不需要额外同步。

常见误用现象:

  • 在 for-each 循环里调用 remove() —— 实际删的是旧副本,当前迭代器仍指向原数组,逻辑出错
  • 把它当普通 ArrayList 用在高频增删场景,CPU 和内存压力陡增
  • 期望它保证「实时一致性」:写入后,正在遍历的旧迭代器看不到新元素,这是设计使然,不是 bug

使用 Collections.synchronizedList() 后为什么还会出现 ConcurrentModificationException

因为 Collections.synchronizedList(new ArrayList()) 只保证单个方法调用是线程安全的,不保证复合操作的原子性。例如 if (list.isEmpty()) list.add(x) 中,isEmpty()add() 之间可能被其他线程插入元素,导致逻辑错误;更严重的是,迭代时若没手动同步,仍会触发 ConcurrentModificationException

正确写法必须显式加锁:

List<String> list = Collections.synchronizedList(new ArrayList<>());
// 迭代必须手动同步
synchronized (list) {
    Iterator<String> it = list.iterator();
    while (it.hasNext()) {
        System.out.println(it.next());
    }
}

关键点:

  • synchronizedList 包装类只是“语法糖”,底层仍依赖用户控制临界区
  • 它无法防止迭代过程中的结构性修改(除非你始终用同步块包裹整个遍历)
  • 相比 CopyOnWriteArrayList,它更适合写操作较少、但需要强一致性读写的场景

BlockingQueue 如何配合线程池实现生产者-消费者解耦

典型组合是 ThreadPoolExecutor + ArrayBlockingQueueLinkedBlockingQueue。线程池的 workQueue 参数就是用来接住提交但暂未执行的任务,本质就是一个阻塞队列。

参数差异影响明显:

  • ArrayBlockingQueue 是有界队列,容量固定,能防止内存溢出,但任务提交超限时会触发线程池的拒绝策略(如 AbortPolicyRejectedExecutionException
  • LinkedBlockingQueue 默认无界(Integer.MAX_VALUE),看似“不会满”,但实际可能导致 OOM;设为有界更可控
  • 别直接用 PriorityBlockingQueue 当线程池队列——它不保证公平性,且任务执行顺序与优先级不总一致

容易忽略的一点:线程池的 corePoolSize 和队列容量共同决定资源水位。例如 corePoolSize=2、队列容量=10,意味着前 12 个任务不会创建新线程;第 13 个才可能触发扩容(取决于 maxPoolSize)。这个组合逻辑常被低估。

到这里,我们也就讲完了《Java同步集合与并发编程全解析》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>