@

一、前言

我们在高并发的场景下,难免会出现并发问题,特别是ArrayList这种常用的集合。这种事情还是要考虑的,今天就带大家一起看一下ArrayList为什么不安全?有哪些解决方案呢?

二、为什么线程不安全

1. 出错演示

import java.util.ArrayList;
import java.util.List;
import java.util.UUID; public class ArrayListDemo { public static void main(String[] args) { List<String> list = new ArrayList<>(); for (int x = 0; x < 20; x++) {
new Thread(() -> {
list.add(UUID.randomUUID().toString());
System.out.println(list);
},String.valueOf(x)).start();
}
}
}

2. ConcurrentModificationException(并发修改异常)



3. 原因

ArrayList 是线程不安全的,多个线程同时操作会出异常 ,并发修改出现异常,导致modCount和操作次数不一致。

//ArrayList源码
private void ensureExplicitCapacity(int minCapacity) {
modCount++; // overflow-conscious code
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}

三、解决方案一CopyOnWriteArrayList (推荐,读多写少场景)

1. 解决方案

import java.util.ArrayList;
import java.util.List;
import java.util.UUID;
import java.util.concurrent.CopyOnWriteArrayList; public class ArrayListDemo { public static void main(String[] args) { List<String> list = new CopyOnWriteArrayList();
for (int x = 0; x < 20; x++) {
new Thread(() -> {
list.add(UUID.randomUUID().toString());
System.out.println(list);
},String.valueOf(x)).start();
}
}
}

2. 源码解读

/**
* 写时复制
* 思想就是先拷贝出来,在拷贝集合的最后添加新元素,最后把集合在set进去
* 写的时候进行加锁,读没有限制
* 适用场景:读多写少
*/
public boolean add(E e) {
final ReentrantLock lock = this.lock;
// 加可重入锁
lock.lock();
try {
// 拷贝当前集合数组
Object[] elements = getArray();
// 获取当前集合数组长度
int len = elements.length;
// 原集合数组空间+1,同时将原集合数组复制
Object[] newElements = Arrays.copyOf(elements, len + 1);
// 把新增元素赋值到新集合数组中
newElements[len] = e;
// 最后再把新集合数组设置到集合中
setArray(newElements);
return true;
} finally {
// 释放锁
lock.unlock();
}
}

四、Collections.synchronizedList(加锁)

1. 解决方案

public class ArrayListDemo {

    public static void main(String[] args) {

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

        for (int x = 0; x < 20; x++) {
new Thread(() -> {
list.add(UUID.randomUUID().toString());
System.out.println(list);
},String.valueOf(x)).start();
}
}
}

2. 源码解读(加锁同步代码块)

public boolean add(E e) {
synchronized (mutex) {return c.add(e);}
}

五、Vector (加锁)

1. 解决方案

public class ArrayListDemo {

    public static void main(String[] args) {

        List<String> list = new Vector<>();

        for (int x = 0; x < 20; x++) {
new Thread(() -> {
list.add(UUID.randomUUID().toString());
System.out.println(list);
},String.valueOf(x)).start();
}
}
}

2. 源码解读(加锁同步方法)

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

六、synchronizedList、Vector都是加锁,区别?

同步代码块和同步方法的区别:



因为SynchronizedList只是使用同步代码块包裹了ArrayList的方法,而ArrayList和Vector中同名方法的方法体内容并无太大差异,所以在锁定范围和锁的作用域上两者并无区别。 在锁定的对象区别上,SynchronizedList的同步代码块锁定的是mutex对象,Vector锁定的是this对象

而其实mutex对象就是SynchronizedList有一个构造函数可以传入一个Object类型对象,如果在调用的时候显示的传入一个对象,那么锁定的就是用户传入的对象。如果没有指定,那么锁定的也是this对象。结果就是SynchronizedList可以指定锁定的对象

基于我们将ArrayList转成SynchronizedList。那么如果我们想把LinkedList变成线程安全的,或者说我想要方便在中间插入和删除的同步的链表,那么我可以将已有的LinkedList直接转成SynchronizedList,而不用改变他的底层数据结构。而这一点是Vector无法做到的,因为他的底层结构就是使用数组实现的,这个是无法更改的



SynchronizedList和Vector区别:

1.SynchronizedList有很好的扩展和兼容功能。他可以将所有的List的子类转成线程安全的类

2.使用SynchronizedList的时候,进行遍历时要手动进行同步处理

3.SynchronizedList可以指定锁定的对象



答案来源

七、总结

这样我们就对为什么不安全和三种解决方案进行测试和源码的初探,小编也是菜鸟,主要是看了尚硅谷阳哥的课。自己记录一遍,以后看!!喜欢的收藏一波,收藏从未停止,行动从未开始!!

集合不安全之 ArrayList及其三种解决方案【CopyOnWriteArrayList 、synchronizedList、Vector 】的更多相关文章

  1. Java集合框架Collection(1)ArrayList的三种遍历方法

    ArrayList是java最重要的数据结构之一,日常工作中经常用到的就是ArrayList的遍历,经过总结,发现大致有三种,上代码: package com.company; import java ...

  2. 【Java集合源代码剖析】ArrayList源代码剖析

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/mmc_maodun/article/details/35568011 转载请注明出处:http:// ...

  3. Java集合源码剖析——ArrayList源码剖析

    ArrayList简介 ArrayList是基于数组实现的,是一个动态数组,其容量能自动增长,类似于C语言中的动态申请内存,动态增长内存. ArrayList不是线程安全的,只能用在单线程环境下,多线 ...

  4. Java 集合系列03之 ArrayList详细介绍(源码解析)和使用示例

    概要 上一章,我们学习了Collection的架构.这一章开始,我们对Collection的具体实现类进行讲解:首先,讲解List,而List中ArrayList又最为常用.因此,本章我们讲解Arra ...

  5. Java学习笔记27(集合框架一:ArrayList回顾、Collection接口方法)

    集合:集合是java中提供的一种容器,可以用来存储多个数据 集合和数组的区别: 1.数组的长度是固定的,集合的长度是可变的 2.集合中存储的元素必须是引用类型数据 对ArrayList集合的回顾 示例 ...

  6. java集合的实现细节--ArrayList和LinkedList

     ArrayList和LinkedList的实现差异 List代表一种线性表的数据结构,ArrayList则是一种顺序存储的线性表,ArrayList底层采用动态数组的形式保存每一个集合元素,Link ...

  7. 集合(一)ArrayList

    前言 这个分类中,将会写写Java中的集合.集合是Java中非常重要而且基础的内容,因为任何数据必不可少的就是该数据是如何存储的,集合的作用就是以一定的方式组织.存储数据.这里写的集合,一部分是比较常 ...

  8. Java集合源代码剖析(一)【集合框架概述、ArrayList、LinkedList、Vector】

    Java集合框架概述 Java集合工具包位于Java.util包下.包括了非常多经常使用的数据结构,如数组.链表.栈.队列.集合.哈希表等.学习Java集合框架下大致能够分为例如以下五个部分:List ...

  9. Java集合系列(二):ArrayList、LinkedList、Vector的使用方法及区别

    本篇博客主要讲解List接口的三个实现类ArrayList.LinkedList.Vector的使用方法以及三者之间的区别. 1. ArrayList使用 ArrayList是List接口最常用的实现 ...

随机推荐

  1. java foreach循环抛出异常java.util.ConcurrentModificationException

    代码如下: for (Iterator<String> iter = list.iterator(); iter.hasNext(); ) { if (Integer.parseInt(i ...

  2. Spring Boot事务支持

    一.创建项目 二.添加依赖 <dependencies> <dependency> <groupId>org.projectlombok</groupId&g ...

  3. 最基础前端路由实现,事件popstate使用

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  4. HTTPS及流程简析

    [序] 在我们在浏览某些网站的时候,有时候浏览器提示需要安装根证书,可是为什么浏览器会提示呢?估计一部分人想也没想就直接安装了,不求甚解不好吗? 那么什么是根证书呢?在大概的囫囵吞枣式的百度之后知道了 ...

  5. hooks中,useState异步问题解决方案

    问题描述: 在hooks中,修改状态的是通过useState返回的修改函数实现的.它的功能类似于class组件中的this.setState().而且,这两种方式都是异步的.可是this.setSta ...

  6. 如何基于 Docker 快速搭建 Springboot + Mysql + Redis 项目

    目录 前言 项目目录 搭建项目 1. docker安装启动mysql以及redis 1.1 安装mysql 1.2 安装redis 2. 初始化数据库 3.创建项目 4.初始化代码 4.1 全局配置文 ...

  7. mysql 连接数过多内存增长过快

    mysql连接允许长连接和短链接,但是本身建立连接会有很大开销所以一般连接mysql会使用长连接.但是全部是使用长连接后,可能会出现mysql有时占用内存涨的特别快,这是因为mysql在执行过程中临时 ...

  8. java 输入输出IO流 字节流| 字符流 的缓冲流:BufferedInputStream;BufferedOutputStream;BufferedReader(Reader in);BufferedWriter(Writer out)

    什么是缓冲流: 缓冲流的基本原理,是在创建流对象时,会创建一个内置的默认大小的缓冲区数组,通过缓冲区读写,减少系统IO次数,从而提高读写的效率. 图解: 1.字节缓冲流BufferedInputStr ...

  9. ELK 使用filebeat替代Logstash收集日志

    使用beats采集日志 之前也介绍过beats是ELK体系中新增的一个工具,它属于一个轻量的日志采集器,以上我们使用的日志采集工具是logstash,但是logstash占用的资源比较大,没有beat ...

  10. 如何在服务器上新建一个svn的目录工程文件

    如何在服务器上新建一个svn的目录工程文件