并发编程之的ArrayList安全性的详细解析

简介: 并发编程之的ArrayList安全性的详细解析

ArrayList不安全

ArrayList中的add方法没有synchronized修饰,是不安全的

下面代码运行结果(异常 java.util.ConcurrentModificationException):



import java.util.ArrayList;
import java.util.List;
import java.util.UUID;
/**
 * @author zkw
 * @Description list不安全
 */
public class ThreadList {
    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        for (int i = 0; i < 30; i++) {
            new Thread(()->{
                list.add(UUID.randomUUID().toString().substring(0,8));
                System.out.println(list);
            }, i+"").start();
        }
    }
}
解决方法
  1. 使用Vector (已过时) 使用的是synchronized机制
  2. 使用Collections.synchronizedList()方法修饰ArrayList (性能不高)
    Collections中还有synchronizedMap和synchronizedSet的方法,可以修饰线程不安全的HashSet,HashMap
  3. 使用java.util.concurrent.CopyOnWriteArrayList(推荐)使用的是lock锁

使用CopyOnWriteArrayList

底层使用的是ReentrantLock可重用锁

下面给出CopyOnWriteArrayList 的 add源码

 

 final transient ReentrantLock lock = new ReentrantLock();    
    /**
     * Appends the specified element to the end of this list.
     *
     * @param e element to be appended to this list
     * @return {@code true} (as specified by {@link Collection#add})
     */
    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;
/**
 * @author zkw
 * @Description list不安全
 */
public class ThreadList {
    public static void main(String[] args) {
        List<String> list = new CopyOnWriteArrayList<>();
        for (int i = 0; i < 30; i++) {
            new Thread(()->{
                list.add(UUID.randomUUID().toString().substring(0,8));
                System.out.println(list);
            }, i+"").start();
        }
    }
}

CopyOnWriteArrayList解析

CopyOnWriteArrayList 利用的是读写分离的思想,读和写的是不同的容器

底层使用的是一个Object数组,每次新添加元素的时候利用的是Arrays.copyOf来创建一个新数组达到扩容效果

使用的是ReentrantLock来保证线程安全

写时复制 Copyonwrite容器即写时复制的容器。往一个容器添加元素的时候,不直接往当前容涨object[]添加,而是先将当前容器 object[ ]进行copy,复制出一个新的容器object[ ] newELements,然后新的容器object[ ] newELements里添加元素,添加 完元素之后, 再将原容器的引用指向新的容器setArray(newELements);。这样做的好处是可以对Copyonwrite容器进行并发的读,而不需 要加锁,因为当前容器不会添加任何元素。所以Copyonwdrite容器也是一种读写分离的思想,读和写不同的容器。

 

 final transient ReentrantLock lock = new ReentrantLock();  
    private transient volatile Object[] array;
    /**
     * Appends the specified element to the end of this list.
     *
     * @param e element to be appended to this list
     * @return {@code true} (as specified by {@link Collection#add})
     */
    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); //将新容器设置为该list的容器
            return true;
        } finally {
            lock.unlock();
        }
    }


相关文章
|
存储 NoSQL 编译器
实战总结|抽丝剥茧,记一次神奇的崩溃
本文详细回放了一个崩溃案例的分析过程。回顾了C++多态和类内存布局、pc指针与芯片异常处理、内存屏障的相关知识。
|
11月前
|
SQL 分布式计算 数据挖掘
加速数据分析:阿里云Hologres在实时数仓中的应用实践
【10月更文挑战第9天】随着大数据技术的发展,企业对于数据处理和分析的需求日益增长。特别是在面对海量数据时,如何快速、准确地进行数据查询和分析成为了关键问题。阿里云Hologres作为一个高性能的实时交互式分析服务,为解决这些问题提供了强大的支持。本文将深入探讨Hologres的特点及其在实时数仓中的应用,并通过具体的代码示例来展示其实际应用。
532 0
|
Java API Spring
Spring Boot 中的 AOP 处理
对 Spring Boot 中的切面 AOP 做了详细的讲解,主要介绍了 Spring Boot 中 AOP 的引入,常用注解的使用,参数的使用,以及常用 api 的介绍。AOP 在实际项目中很有用,对切面方法执行前后都可以根据具体的业务,做相应的预处理或者增强处理,同时也可以用作异常捕获处理,可以根据具体业务场景,合理去使用 AOP。
|
存储 Go
GO通道:无缓冲通道与缓冲通道
GO通道:无缓冲通道与缓冲通道
105 0
|
Dubbo Java 应用服务中间件
走向 Native 化:Spring&Dubbo AOT 技术示例与原理讲解
走向 Native 化:Spring&Dubbo AOT 技术示例与原理讲解
798 1
|
专有云
一张图看懂专有云灾备
政企为什么要做专有云灾备?灾备的目标是什么?如何基于业务需求,选择合适的灾备方案?
1011 0
一张图看懂专有云灾备
|
运维 监控 数据挖掘
带你读《2022技术人的百宝黑皮书》——大淘宝技术数据模型治理阶段性分享(2)
带你读《2022技术人的百宝黑皮书》——大淘宝技术数据模型治理阶段性分享(2)
203 0
|
人工智能 安全 搜索推荐
人脸识别在当今世界的重要性是什么?
鉴于全球范围内的 COVID-19 流行病,我们可能会期待对面部识别等生物识别技术进行更多投资。
人脸识别在当今世界的重要性是什么?
|
SQL druid 关系型数据库
Seata AT 分支事务
前面,我们已经介绍了 Seata 的整体设计思想,接下来我们深入到其实现细节中,本文介绍 Seata 中 AT 模式分支事务的实现。
|
Oracle 关系型数据库 Linux
Linux系统中Oracle数据库使用SELECT语句检索数据(1)实例应用
1,首先切换到Oracle用户,并进入数据库#sql / as sysdba2,启动数据库,并连接样例及表格,启动命令#startup,连接样例#conn scott/tiger3,select语句中:不区分大小写;可以写一行或多行,为方便查看最好每个子句单独一行;语句以“;”结尾结束语句4,se.