有一天,我和 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 啦!

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

相关推荐
咩图5 分钟前
Sketchup软件二次开发+Ruby+VisualStudioCode
java·前端·ruby
我命由我123458 分钟前
Android 开发问题:Duplicate class android.support.v4.app.INotificationSideChannel...
android·java·开发语言·java-ee·android studio·android-studio·android runtime
熬夜敲代码的小N14 分钟前
Unity大场景卡顿“急救包”:从诊断到落地的全栈优化方案
java·unity·游戏引擎
坚持学习前端日记15 分钟前
容器化中间件的优缺点
java·中间件
黑客老李16 分钟前
一次有趣的通杀
java·数据库·mysql
季明洵20 分钟前
反转字符串、反转字符串II、反转字符串中的单词
java·数据结构·算法·leetcode·字符串
虫小宝26 分钟前
查券返利机器人的异步任务调度:Java XXL-Job+Redis实现海量查券请求的分布式任务分发
java·redis·分布式
Mr_Xuhhh37 分钟前
C语言字符串与内存操作函数模拟实现详解
java·linux·算法
瑞雪兆丰年兮38 分钟前
[从0开始学Java|第十一天]ArrayList
java·开发语言
夜郎king42 分钟前
基于 Java 实现数九天精准计算:从节气算法到工程化落地
java·开发语言