- ArrayList 线程不安全会造成 java.util.ConcurrentModificationException(并发争抢修改导致异常),示例
private static void list() {
List<String> list = new ArrayList<>();
List<String> list2 = new Vector<>();
List<String> list3 = Collections.synchronizedList(new ArrayList<>());
List<String> list4 = new CopyOnWriteArrayList<>();
for (int i = 0; i <= 30; i++) {
new Thread(() -> {
list.add(UUID.randomUUID().toString().substring(0, 8));
System.out.println(list);
}, String.valueOf(i)).start();
}
}
private static void set() {
Set<String> set = new HashSet<>();
Set<String> set2 = Collections.synchronizedSet(new HashSet<>());
Set<String> set3 = new CopyOnWriteArraySet<>();
for (int i = 0; i <= 30; i++) {
new Thread(() -> {
set.add(UUID.randomUUID().toString().substring(0, 8));
System.out.println(set);
}, String.valueOf(i)).start();
}
}
private static void map() {
Map<String, String> map = new HashMap<>();
Map<String, String> map2 = new ConcurrentHashMap<>();
Map<String, String> map3 = Collections.synchronizedMap(new HashMap<>());
for (int i = 0; i <= 30; i++) {
new Thread(() -> {
map3.put(Thread.currentThread().getName(), UUID.randomUUID().toString().substring(0, 8));
System.out.println(map3);
}, String.valueOf(i)).start();
}
}
解决方式
a.List list = new Vector<>()
该方法是synchronized,但是比较out,数据一致性可以保证,但是并发性急剧下降
b.List list1 = Collections.synchronizedList(new ArrayList<>())
(***SynchronizedList和Vector最主要的区别:总的来说,V给方法加锁,S给代码块加锁,所以S效率高。 1.SynchronizedList有很好的扩展和兼容功能。他可以将所有的List的子类转成线程安全的类。 2.使用SynchronizedList的时候,进行遍历时要手动进行同步处理。3.SynchronizedList可以指定锁定的对象。)
c.List list2 = new CopyOnWriteArrayList<>();
写时复制:先复制再写值,再返回新数组。该过程用ReentrantLock锁住。