有一天,我和 CopyOnWriteArrayList 杯“线程安全”的咖啡

原文来自于:[zha-ge.cn/java/38]zha-ge.cn/java/38)

有一天,我和 CopyOnWriteArrayList 杯"线程安全"的咖啡

"说出来你可能不信,我第一次遇见 CopyOnWriteArrayList,真有点像突然喝到了一杯奇葩口味咖啡。" 那天项目里一顿多线程操作 ArrayList,直接就炸了------各种 ConcurrentModificationException 扑面而来。眼看需求日期在逼近,我只能抱着百度拼命啃资料,终于,那个神秘的名字------CopyOnWriteArrayList,出现在我的面前。

奇怪的名字,神奇的用法

这货的名字一看就不走寻常路:"写时复制"列表。跟平时 ArrayList 那种你加我加大家抢着改的作风完全不一样。 CopyOnWriteArrayList 的大法简单粗暴:只要有修改操作,比如 add 或 remove,它就直接把底层的数组"复制"一份,改完再换上。 读的时候,全世界读线程用的都是一个老版本的新数组,根本不用加锁。 写的时候,悄咪咪整出个新副本,等你们都读完了再说。

代码长这样:

java 复制代码
CopyOnWriteArrayList<String> list = new CopyOnWriteArrayList<>();
list.add("程序员");
list.add("出门左转");
list.remove("左转");
// ...

嗯,是不是看着很清爽? 连 iterator 都不用担心:

java 复制代码
for (String s : list) {
    System.out.println(s);
    // list.add("插队"); // 不会抛 ConcurrentModificationException
}

想当初用普通 ArrayList,改着改着分分钟爆炸。CopyOnWriteArrayList 则"稳如老狗"。

踩坑瞬间

说起来简单,真用的时候,坑也是多得想抽自己耳光。

  • 那天真有个需求,存个8万条数据要频繁 add/remove。本想 CopyOnWriteArrayList 线程安全,直接莽,结果垃圾回收疯狂爆发,内存飙升。
  • 遍历过程中,想试试能不能边遍历边加元素,嗯,是不会抛异常,结果新加的东西遍历完了也看不到......
  • 查文档的时候脑子一热:既然线程安全,啥场面都能用吧!后来 Leader 只说了三个字:"开除吧!"

经验启示

时间一久,我总结出一套"用 CopyOnWriteArrayList 的锦囊":

  • 适合读多写少场景 频繁写?直接 gg,别用这玩意儿,损失性能简直肉眼可见。
  • 遍历期间修改,遍历不到最新元素 它的 iterator 永远"活在过去"。要最新数据,遍历得重来一遍。
  • 内存占用高 千万别往里塞上万条、甚至百万级别数据。
  • 线程安全≠银弹 要是真有写多场景,还得老老实实用锁或者选用别的并发集合。

适合这样:

  • "配置黑板"类读多改少
  • 维护小容量缓存
  • 偶尔边迭代边刮胡子改几个元素

收个尾巴

说白了,CopyOnWriteArrayList 就是个谈恋爱怕闹分手的"安全派": 你要改,我就自个儿带球跑路,剩下的你们慢慢看前任。 用得好,项目稳如老狗;用错场景,队友只会拿你祭天。

所以,下回再遇到多线程需要安全集合的场景,还得盘一盘,别再头铁乱撸 CopyOnWriteArrayList 啦!

------聊着聊着,咖啡快凉了,晚安,我的并发朋友们。

相关推荐
叽哥7 小时前
Kotlin学习第 3 课:Kotlin 流程控制:掌握逻辑分支与循环的艺术
android·java·kotlin
杨杨杨大侠7 小时前
第5章:实现Spring Boot集成
java·github·eventbus
华仔啊7 小时前
工作5年没碰过分布式锁,是我太菜还是公司太稳?网友:太真实了!
java·后端
尚久龙7 小时前
安卓学习 之 图片控件和图片按钮
android·java·学习·手机·android studio·安卓
摸鱼仙人~7 小时前
深入理解 MyBatis-Plus 的 `BaseMapper`
java·开发语言·mybatis
杨杨杨大侠7 小时前
第6章:高级特性与性能优化
java·github·eventbus
Dcs7 小时前
代码评审还能更好!
java
刃神太酷啦8 小时前
C++ 异常处理机制:从基础到实践的全面解析----《Hello C++ Wrold!》(20)--(C/C++)
java·c语言·开发语言·c++·qt·算法·leetcode
蓝倾9768 小时前
小红书获取用户作品列表API接口操作指南
java·服务器·前端·python·电商开放平台·开放api接口