为了账号安全,请及时绑定邮箱和手机立即绑定

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

标签:
Java JavaScript

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

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

奇怪的名字,神奇的用法

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

代码长这样:

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

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

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

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

踩坑瞬间

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

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

经验启示

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

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

适合这样:

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

收个尾巴

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

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

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

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号

举报

0/150
提交
取消