集合线程安全问题:第一章:集合类不安全之并发修改异常

本文深入探讨了ArrayList、HashSet和HashMap在多线程环境下的线程安全性问题,分析了java.util.ConcurrentModificationException异常的产生原因,并提供了三种解决策略,包括使用Vector、Collections.synchronized和CopyOnWrite系列集合。

直接上ArrayList线程不安全代码:

package com.javaliao.backstage;

import java.util.ArrayList;
import java.util.UUID;

public class Demo {
    public static void main(String[] args) {
        List arrayList = new ArrayList<String>();
        for (int i = 0; i < 30; i++) {
            new Thread(()->{
                arrayList.add(UUID.randomUUID().toString().substring(0,8));
                System.out.println(arrayList);
            },String.valueOf(i)).start();
        }
    }
}

控制台直接报错:

只要你干过电商项目的基本上都见过,java.util.ConcurrentModificationException并发修改异常

错误分析:

  • 故障现象:java.util.ConcurrentModificationException并发修改异常
  • 导致原因:并发争取修改导致,一个线程正在写,一个线程过来争抢,导致线程写的过程被其他线程打断,导致数据不一致。
  • 解决方案

          第一种:使用List arrayList = new Vector<>();它的底层使用了synchronized加锁,但是并发下降

          第二种:使用List arrayList = Collections.synchronizedList(new ArrayList<String>());使用工具类,线程同步

          第三种:使用List arrayList = new CopyOnWriteArrayList<>();写时复制

  • 优化建议

使用第三种解决方案较好

直接上HashSet线程不安全代码:

package com.javaliao.backstage;

import java.util.*;

public class Demo {
    public static void main(String[] args) {
        Set hashSet = new HashSet<String>();
        for (int i = 0; i < 30; i++) {
            new Thread(()->{
                hashSet.add(UUID.randomUUID().toString().substring(0,8));
                System.out.println(hashSet);
            },String.valueOf(i)).start();
        }
    }
}

控制台:

错误分析: 

  • 故障现象:java.util.ConcurrentModificationException并发修改异常
  • 导致原因:并发争取修改导致,一个线程正在写,一个线程过来争抢,导致线程写的过程被其他线程打断,导致数据不一致。
  • 解决方案

        第一种:Set<String> hashSet = Collections.synchronizedSet(new HashSet<>());

        第二种:Set<String> hashSet = new CopyOnWriteArraySet();(它的底层还是 new CopyOnWriteArrayList<>();)

直接上HashMap线程不安全代码:

package com.javaliao.backstage;

import java.util.*;

public class Demo {
    public static void main(String[] args) {
        Map<String,String> hashMap = new HashMap<>();
        for (int i = 0; i < 30; i++) {
            new Thread(()->{
                hashMap.put(Thread.currentThread().getName(),UUID.randomUUID().toString().substring(0,8));
                System.out.println(hashMap);
            },String.valueOf(i)).start();
        }
    }
}

控制台:

错误分析: 

  • 故障现象:java.util.ConcurrentModificationException并发修改异常
  • 导致原因:并发争取修改导致,一个线程正在写,一个线程过来争抢,导致线程写的过程被其他线程打断,导致数据不一致。
  • 解决方案:

             第一种:Map<String,String> hashMap = new ConcurrentHashMap<>();

             第二种:Map<String,String> hashMap = Collections.synchronizedMap(new HashMap<>());

 

 

 

 

 

 

 

 

 

 

 

 

评论 38
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Java程序员廖志伟

赏我包辣条呗

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值