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

2022-09-28 17:00:34 浏览数 (1)

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

代码语言:javascript复制
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线程不安全代码:

代码语言:javascript复制
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线程不安全代码:

代码语言:javascript复制
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<>());

0 人点赞