ArrayList为什么线程不安全以及三种解决办法【详细】
创始人
2024-03-16 19:45:00

目录

  • 不安全原因
  • 解决办法
    • Vector
    • Collections
    • CopyOnWriteArrayList
  • 三种解决方式总结

不安全原因

  • 我们可以看一下ArrayList源码,找到add方法,
public boolean add(E e) {ensureCapacityInternal(size + 1);  // Increments modCount!!elementData[size++] = e;return true;
}

从上面的代码可以看出,add()方法没有使用同步互斥,所以在多线程并发中,会出现线程异常,测试代码:

import java.util.ArrayList;
import java.util.UUID;public class SetUnsefertyTest {public static void main(String[] args) {// 创建ArrayList 集合ArrayList list = new ArrayList<>();// 创建10个线程,往 list 中添加元素for (int i = 0; i < 10; i++) {new Thread(()->{// 向集合中添加内容list.add(UUID.randomUUID().toString().substring(0,8));// 从集合中取出内容System.out.println(list);},String.valueOf(i)).start();}}
}

出现异常:
请添加图片描述

解决办法

Vector

可以看一下Vector的add源码,加上了synchronized同步关键字
但是 Vector 用的不多,因为每次对添加的元素上锁,而且使用的是重量级锁synchronized是十分占用资源的,效率是十分低下的。其用法和 ArrayList 一样。

public synchronized boolean add(E e) {modCount++;ensureCapacityHelper(elementCount + 1);elementData[elementCount++] = e;return true;
}

测试代码:

import java.util.ArrayList;
import java.util.List;
import java.util.UUID;
import java.util.Vector;public class SetUnsefertyTest {public static void main(String[] args) {// 创建ArrayList 集合List list = new Vector();// 创建10个线程,往 list 中添加元素for (int i = 0; i < 10; i++) {new Thread(()->{// 向集合中添加内容list.add(UUID.randomUUID().toString().substring(0,8));// 从集合中取出内容System.out.println(list);},String.valueOf(i)).start();}}
}

Collections

进入 Collections 的底层,找到 synchronizedList(List list) 方法,源代码如下,synchronizedList(List list) 方法返回指定列表支持的同步(线程安全的)列表

public static  List synchronizedList(List list) {return (list instanceof RandomAccess ?new SynchronizedRandomAccessList<>(list) :new SynchronizedList<>(list));
}static  List synchronizedList(List list, Object mutex) {return (list instanceof RandomAccess ?new SynchronizedRandomAccessList<>(list, mutex) :new SynchronizedList<>(list, mutex));
}

对 Collections 的使用如下

List list = Collections.synchronizedList(new ArrayList<>());

测试代码:

import java.util.*;public class SetUnsefertyTest {public static void main(String[] args) {// 创建ArrayList 集合
//        List list = new Vector();List list = Collections.synchronizedList(new ArrayList<>()) ;// 创建10个线程,往 list 中添加元素for (int i = 0; i < 10; i++) {new Thread(()->{// 向集合中添加内容list.add(UUID.randomUUID().toString().substring(0,8));// 从集合中取出内容System.out.println(list);},String.valueOf(i)).start();}}
}

CopyOnWriteArrayList

这是写时复制思想, 首先看add()方法中有可重入锁,这个目的是防止多个线程争抢写的权力,然后下面红框中的内容是将原件复制出来一份,然后在复印件上写,之后通过setArray()方法让原件地址指向复印件,这样可以让所有人读原件,而我只修改复印件,所以读和写不会出现冲突,因此通过加锁和写时复制思想可以很好保证了多线程情况下所有线程都可以读,但是只有一个线程在写,因此不会出现并发修改异常,如源码:

public boolean add(E e) {// 声明一个重进入锁final ReentrantLock lock = this.lock;// 上锁lock.lock();try {// 获取原来的列表Object[] elements = getArray();// 获取原来列表的长度int len = elements.length;// 复制一个与原来的列表一样的列表Object[] newElements = Arrays.copyOf(elements, len + 1);// 将新加入的元素放到列表末尾newElements[len] = e;// 旧新合并setArray(newElements);return true;} finally {// 解锁lock.unlock();}
}

测试代码:

import java.util.*;
import java.util.concurrent.CopyOnWriteArrayList;public class SetUnsefertyTest {public static void main(String[] args) {// 创建ArrayList 集合
//        List list = new Vector();//        List list = Collections.synchronizedList(new ArrayList<>()) ;List list =new CopyOnWriteArrayList<>() ;// 创建10个线程,往 list 中添加元素for (int i = 0; i < 10; i++) {new Thread(()->{// 向集合中添加内容list.add(UUID.randomUUID().toString().substring(0,8));// 从集合中取出内容System.out.println(list);},String.valueOf(i)).start();}}
}

三种解决方式总结

对比三者来看,Vector和Collections虽然也可以实现同步,但由于这两种方法在底层都使用了synchronized重量级锁,使其效率很低,所以对 ArrayList 的同步主要采用 CopyOnWriteArrayList

相关内容

热门资讯

埃菲尔铁塔在哪 中国仿建埃菲尔... 2019年4月26日,广西南宁市,街头惊现一座巨型山寨版埃菲尔铁塔,高约20米,白色塔身,造型逼真,...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...
脚上的穴位图 脚面经络图对应的... 人体穴位作用图解大全更清晰直观的标注了各个人体穴位的作用,包括头部穴位图、胸部穴位图、背部穴位图、胳...
应用未安装解决办法 平板应用未... ---IT小技术,每天Get一个小技能!一、前言描述苹果IPad2居然不能安装怎么办?与此IPad不...
世界上最漂亮的人 世界上最漂亮... 此前在某网上,选出了全球265万颜值姣好的女性。从这些数量庞大的女性群体中,人们投票选出了心目中最美...
猫咪吃了塑料袋怎么办 猫咪误食... 你知道吗?塑料袋放久了会长猫哦!要说猫咪对塑料袋的喜爱程度完完全全可以媲美纸箱家里只要一有塑料袋的响...
demo什么意思 demo版本... 618快到了,各位的小金库大概也在准备开闸放水了吧。没有小金库的,也该向老婆撒娇卖萌服个软了,一切只...
埃菲尔铁塔在哪 中国仿建埃菲尔... 2019年4月26日,广西南宁市,街头惊现一座巨型山寨版埃菲尔铁塔,高约20米,白色塔身,造型逼真,...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...