时间:2021-05-20
1.ArrayList源码和多线程安全问题分析
在分析ArrayList线程安全问题之前,我们线对此类的源码进行分析,找出可能出现线程安全问题的地方,然后代码进行验证和分析。
1.1 数据结构
ArrayList内部是使用数组保存元素的,数据定义如下:
transient Object[] elementData; // non-private to simplify nested class access在ArrayList中此数组即是共享资源,当多线程对此数据进行操作的时候如果不进行同步控制,即有可能会出现线程安全问题。
1.2 add方法可能出现的问题分析
首先我们看一下add的源码如下:
public boolean add(E e) {ensureCapacityInternal(size + 1);elementData[size++] = e;return true;}此方法中有两个操作,一个是数组容量检查,另外就是将元素放入数据中。我们先看第二个简单的开始分析,当多个线程执行顺序如下所示的时候,会出现最终数据元素个数小于期望值。
按照此顺序执行完之后,我们可以看到,elementData[n]的只被设置了两次,第二个线程设置的值将前一个覆盖,最后size=n+1。下面使用代码进行验证此问题。
1.3 代码验证
首先先看下以下代码,开启1000个线程,同时调用ArrayList的add方法,每个线程向ArrayList中添加100个数字,如果程序正常执行的情况下应该是输出:
list size is :10000
代码如下:
private static List<Integer> list = new ArrayList<Integer>();private static ExecutorService executorService = Executors.newFixedThreadPool(1000);private static class IncreaseTask extends Thread{@Overridepublic void run() {System.out.println("ThreadId:" + Thread.currentThread().getId() + " start!");for(int i =0; i < 100; i++){list.add(i);}System.out.println("ThreadId:" + Thread.currentThread().getId() + " finished!");}}public static void main(String[] args){for(int i=0; i < 1000; i++){executorService.submit(new IncreaseTask());}executorService.shutdown();while (!executorService.isTerminated()){try {Thread.sleep(1000*10);}catch (InterruptedException e){e.printStackTrace();}}System.out.println("All task finished!");System.out.println("list size is :" + list.size());}当执行此main方法后,输出如下:
从以上执行结果来看,最后输出的结果会小于我们的期望值。即当多线程调用add方法的时候会出现元素覆盖的问题。
1.4 数组容量检测的并发问题
在add方法源码中,我们看到在每次添加元素之前都会有一次数组容量的检测,add中调用此方法的源码如下:
ensureCapacityInternal(size + 1);
容量检测的相关源码如下:
private void ensureCapacityInternal(int minCapacity) {if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);}ensureExplicitCapacity(minCapacity);}private void ensureExplicitCapacity(int minCapacity) {modCount++;// overflow-conscious codeif (minCapacity - elementData.length > 0)grow(minCapacity);}容量检测的流程图如下所示:
我们以两个线程执行add操作来分析扩充容量可能会出现的并发问题:
当我们新建一个ArrayList时候,此时内部数组容器的容量为默认容量10,当我们用两个线程同时添加第10个元素的时候,如果出现以下执行顺序,可能会抛出java.lang.ArrayIndexOutOfBoundsException异常。
第二个线程往数组中添加数据的时候由于数组容量为10,而此操作往index为10的位置设置元素值,因此会抛出数组越界异常。
1.5 代码验证数组容量检测的并发问题
使用如下代码:
private static List<Integer> list = new ArrayList<Integer>(3);private static ExecutorService executorService = Executors.newFixedThreadPool(10000);private static class IncreaseTask extends Thread{@Overridepublic void run() {System.out.println("ThreadId:" + Thread.currentThread().getId() + " start!");for(int i =0; i < 1000000; i++){list.add(i);}System.out.println("ThreadId:" + Thread.currentThread().getId() + " finished!");}}public static void main(String[] args){new IncreaseTask().start();new IncreaseTask().start();}执行main方法后,我们可以看到控制台输出如下:
1.6 ArrayList中其他方法说明
ArrayList中其他包含对共享变量操作的方法同样会有并发安全问题,只需要按照以上的分析方法分析即可。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
本文主要接着前面多线程的两篇文章总结Java多线程中的线程安全问题。一.一个典型的Java线程安全例子publicclassThreadTest{publics
前言多线程的线程安全问题是微妙而且出乎意料的,因为在没有进行适当同步的情况下多线程中各个操作的顺序是不可预期的,多线程访问同一个共享变量特别容易出现并发问题,特
一.什么时候会出现线程安全问题? 在单线程中不会出现线程安全问题,而在多线程编程中,有可能会出现同时访问同一个资源的情况,这种资源可以是各种类型的的资源:一个
一、线程安全问题产生的原因线程安全问题都是由全局变量及静态变量引起的二、线程安全问题SimpleDateFormatesdf=newSimpleDateForm
Java多线程线程同步与死锁1.线程同步多线程引发的安全问题一个非常经典的案例,银行取钱的问题。假如你有一张银行卡,里面有5000块钱,然后你去银行取款2000