ArrayList为什么线程不安全以及三种解决办法【详细】
创始人
2024-04-16 22:26:50
0

目录

  • 不安全原因
  • 解决办法
    • 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

相关内容

热门资讯

银河麒麟V10SP1高级服务器... 银河麒麟高级服务器操作系统简介: 银河麒麟高级服务器操作系统V10是针对企业级关键业务...
【NI Multisim 14...   目录 序言 一、工具栏 🍊1.“标准”工具栏 🍊 2.视图工具...
AWSECS:访问外部网络时出... 如果您在AWS ECS中部署了应用程序,并且该应用程序需要访问外部网络,但是无法正常访问,可能是因为...
不能访问光猫的的管理页面 光猫是现代家庭宽带网络的重要组成部分,它可以提供高速稳定的网络连接。但是,有时候我们会遇到不能访问光...
AWSElasticBeans... 在Dockerfile中手动配置nginx反向代理。例如,在Dockerfile中添加以下代码:FR...
Android|无法访问或保存... 这个问题可能是由于权限设置不正确导致的。您需要在应用程序清单文件中添加以下代码来请求适当的权限:此外...
月入8000+的steam搬砖... 大家好,我是阿阳 今天要给大家介绍的是 steam 游戏搬砖项目,目前...
​ToDesk 远程工具安装及... 目录 前言 ToDesk 优势 ToDesk 下载安装 ToDesk 功能展示 文件传输 设备链接 ...
北信源内网安全管理卸载 北信源内网安全管理是一款网络安全管理软件,主要用于保护内网安全。在日常使用过程中,卸载该软件是一种常...
AWS管理控制台菜单和权限 要在AWS管理控制台中创建菜单和权限,您可以使用AWS Identity and Access Ma...