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

相关内容

热门资讯

保存时出现了1个错误,导致这篇... 当保存文章时出现错误时,可以通过以下步骤解决问题:查看错误信息:查看错误提示信息可以帮助我们了解具体...
汇川伺服电机位置控制模式参数配... 1. 基本控制参数设置 1)设置位置控制模式   2)绝对值位置线性模...
不能访问光猫的的管理页面 光猫是现代家庭宽带网络的重要组成部分,它可以提供高速稳定的网络连接。但是,有时候我们会遇到不能访问光...
表格中数据未显示 当表格中的数据未显示时,可能是由于以下几个原因导致的:HTML代码问题:检查表格的HTML代码是否正...
本地主机上的图像未显示 问题描述:在本地主机上显示图像时,图像未能正常显示。解决方法:以下是一些可能的解决方法,具体取决于问...
表格列调整大小出现问题 问题描述:表格列调整大小出现问题,无法正常调整列宽。解决方法:检查表格的布局方式是否正确。确保表格使...
不一致的条件格式 要解决不一致的条件格式问题,可以按照以下步骤进行:确定条件格式的规则:首先,需要明确条件格式的规则是...
Android|无法访问或保存... 这个问题可能是由于权限设置不正确导致的。您需要在应用程序清单文件中添加以下代码来请求适当的权限:此外...
【NI Multisim 14...   目录 序言 一、工具栏 🍊1.“标准”工具栏 🍊 2.视图工具...
银河麒麟V10SP1高级服务器... 银河麒麟高级服务器操作系统简介: 银河麒麟高级服务器操作系统V10是针对企业级关键业务...