直接上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<>());