关于ArrayList的越界问题?
大家都知道 ArrayList是自动扩容的。 那为什么会存在越界问题?
话不多说 上代码
package test;
import java.util.ArrayList;
public class ThreadUnSafe {
public static ArrayList<Integer> numberList= new ArrayList<Integer>();
public static class addToList implements Runnable{
int startNum=0;
public addToList(int startNum){
this.startNum=startNum;
}
@Override
public void run(){
int count=0;
while (count<50){
try{
Thread.sleep(100);
}catch (InterruptedException e){
e.printStackTrace();
}
numberList.add(startNum);
System.out.println(Thread.currentThread().getName()+"=="+"第"+(count+1)+"次进入,添加的数子为"+numberList.get(numberList.size()-1)+"---此时集合大小为:"+numberList.size());
startNum+=2;
count++;
}
}
}
public static void main(String[] args) throws InterruptedException{
Thread t1=new Thread(new addToList(0));
Thread t2=new Thread(new addToList(1));
t1.start();
t2.start();
}
}
测试结果:
Thread-1==第1次进入,添加的数字为1---此时集合大小为:1
Thread-0==第1次进入,添加的数字为1---此时集合大小为:1
Thread-0==第2次进入,添加的数字为1---此时集合大小为:2
Thread-1==第2次进入,添加的数字为2---此时集合大小为:3
Thread-0==第3次进入,添加的数字为2---此时集合大小为:4
Thread-1==第3次进入,添加的数字为3---此时集合大小为:5
Thread-0==第4次进入,添加的数字为4---此时集合大小为:7
Thread-1==第4次进入,添加的数字为4---此时集合大小为:7
Thread-0==第5次进入,添加的数字为4---此时集合大小为:8
Thread-1==第5次进入,添加的数字为5---此时集合大小为:9
Thread-1==第6次进入,添加的数字为6---此时集合大小为:10
Thread-0==第6次进入,添加的数字为6---此时集合大小为:10
Thread-0==第7次进入,添加的数字为6---此时集合大小为:11
Thread-1==第7次进入,添加的数字为7---此时集合大小为:12
Thread-1==第8次进入,添加的数字为8---此时集合大小为:14
Thread-0==第8次进入,添加的数字为8---此时集合大小为:14
Thread-1==第9次进入,添加的数字为9---此时集合大小为:15
Thread-0==第9次进入,添加的数字为8---此时集合大小为:16
Thread-1==第10次进入,添加的数字为10---此时集合大小为:17
Thread-0==第10次进入,添加的数字为9---此时集合大小为:18
Thread-0==第11次进入,添加的数字为10---此时集合大小为:19
Thread-1==第11次进入,添加的数字为11---此时集合大小为:20
Thread-0==第12次进入,添加的数字为11---此时集合大小为:21
Thread-1==第12次进入,添加的数字为11---此时集合大小为:21
Thread-1==第13次进入,添加的数字为13---此时集合大小为:22
Thread-0==第13次进入,添加的数字为12---此时集合大小为:23
Thread-1==第14次进入,添加的数字为14---此时集合大小为:25
Thread-0==第14次进入,添加的数字为14---此时集合大小为:25
Thread-1==第15次进入,添加的数字为15---此时集合大小为:26
Thread-0==第15次进入,添加的数字为15---此时集合大小为:26
Thread-0==第16次进入,添加的数字为15---此时集合大小为:28
Thread-1==第16次进入,添加的数字为15---此时集合大小为:28
Thread-1==第17次进入,添加的数字为16---此时集合大小为:29
Thread-0==第17次进入,添加的数字为16---此时集合大小为:29
Thread-0==第18次进入,添加的数字为18---此时集合大小为:31
Thread-1==第18次进入,添加的数字为18---此时集合大小为:31
Thread-0==第19次进入,添加的数字为18---此时集合大小为:32
Thread-1==第19次进入,添加的数字为18---此时集合大小为:32
Exception in thread "Thread-0" Exception in thread "Thread-1" java.lang.ArrayIndexOutOfBoundsException: 33
at java.util.ArrayList.elementData(ArrayList.java:422)
at java.util.ArrayList.get(ArrayList.java:435)
at test.ThreadUnSafe$addToList.run(ThreadUnSafe.java:22)
at java.lang.Thread.run(Thread.java:748)
java.lang.ArrayIndexOutOfBoundsException: 33
at java.util.ArrayList.add(ArrayList.java:463)
at test.ThreadUnSafe$addToList.run(ThreadUnSafe.java:21)
at java.lang.Thread.run(Thread.java:748)
? 为什么会有数组越界呢 。对于ArrayList而言,它实现List接口、底层使用数组保存所有元素。其操作基本上是对数组的操作。
其中:at java.util.ArrayList.add(ArrayList.java:463)的源代码
public boolean add(E e) {
ensureCapacityInternal(size + 1); // Increments modCount!!
elementData[size++] = e;
return true;
}
private void ensureCapacityInternal(int minCapacity) {
ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}
private void ensureExplicitCapacity(int minCapacity) {
modCount++;
// overflow-conscious code
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}
函数体中,modCount是数组发生size更改的次数。然后if判断,如果数组长度小于默认的容量10,则调用扩大数组大小的方法grow()。 其中 函数grow()解释了基于数组的ArrayList是如何扩容的。数组进行扩容时,会将老数组中的元素重新拷贝一份到新的数组中
每次数组容量的增长大约是其原容量的1.5倍。
接下来回到Add()函数,继续执行,elementData[size++] = e; 这行代码就是问题所在,当添加一个元素的时候,它可能会有两步来完成:
1. 在 elementData[Size] 的位置存放此元素;
2. 增大 Size 的值。 在单线程运行的情况下,如果 Size = 0,添加一个元素后,此元素在位置 0,而且 Size=1;
如果是在多线程情况下,比如有两个线程,线程 A 先将元素存放在位置 0。但是此时 CPU 调度线程A暂停,线程 B 得到运行的机会。线程B也向此 ArrayList 添加元素,因为此时 Size 仍然等于 0 (注意哦,我们假设的是添加一个元素是要两个步骤哦,而线程A仅仅完成了步骤1),所以线程B也将元素存放在位置0。然后线程A和线程B都继续运行,都增加 Size 的值。那好,我们来看看 ArrayList 的情况,元素实际上只有一个,存放在位置 0,而 Size 却等于 2。这就是“线程不安全”了
我猜想是,由于没有该方法没有同步,导致出现这样一种现象,用第一次异常,即下标为15时的异常举例。当集合中已经添加了14个元素时,一个线程率先进入add()方法,在执行ensureCapacityInternal(size + 1)时,发现还可以添加一个元素,故数组没有扩容,但随后该线程被阻塞在此处。接着另一线程进入add()方法,执行ensureCapacityInternal(size + 1),由于前一个线程并没有添加元素,故size依然为14,依然不需要扩容,所以该线程就开始添加元素,使得size++,变为15,数组已经满了。而刚刚阻塞在elementData[size++] = e;语句之前的线程开始执行,它要在集合中添加第16个元素,而数组容量只有15个,所以就发生了数组下标越界异常!
关于ArrayList的越界问题?的更多相关文章
- IndexOutOfBoundsException ArrayList 访问越界
java.lang.IndexOutOfBoundsException: Index: 3, Size: 2
- JAVA - ArrayList是否会越界?
JAVA - ArrayList是否会越界? ArrayList并发add()可能出现数组下标越界异常. ArrayList是实现了基于动态数组的数据结构. LinkedList是基于链表的数据结构 ...
- 深入理解java中的ArrayList和LinkedList
杂谈最基本数据结构--"线性表": 表结构是一种最基本的数据结构,最常见的实现是数组,几乎在每个程序每一种开发语言中都提供了数组这个顺序存储的线性表结构实现. 什么是线性表? 由0 ...
- ArrayList LinkedList源码解析
在java中,集合这一数据结构应用广泛,应用最多的莫过于List接口下面的ArrayList和LinkedList; 我们先说List, public interface List<E> ...
- ArrayList源码阅读笔记(基于JDk1.8)
关键常量: private static final int DEFAULT_CAPACITY = 10; 当没有其他参数影响数组大小时的默认数组大小 private static final Obj ...
- java8 ArrayList源码阅读
转载自 java8 ArrayList源码阅读 本文基于jdk1.8 JavaCollection库中有三类:List,Queue,Set 其中List,有三个子实现类:ArrayList,Vecto ...
- java提高篇(二一)-----ArrayList
一.ArrayList概述 ArrayList是实现List接口的动态数组,所谓动态就是它的大小是可变的.实现了所有可选列表操作,并允许包括 null 在内的所有元素.除了实现 List ...
- Java 集合 — ArrayList
ArrayList ArrayList是基于数组实现的List 是有序的 每次添加之前判断是否进行扩容 不是线程安全的. 构造方法 // 空数组 private static final Object ...
- jdk源码分析之ArrayList
ArrayList关键属性分析 ArrayList采用Object数组来存储数据 /** * The array buffer into which the elements of the Array ...
随机推荐
- Django框架——基础之模板系统(template文件夹)
---恢复内容开始--- 1. 常用语法 需要记住两组特殊符号:{{ }} 和 {% %}. 在运用到变量的时候使用{{ }},如果是跟逻辑相关的话就使用{% %}. 在Django模板(t ...
- Swift(一)语言介绍
Swift是苹果2014年推出的全新的编程语言,它继承了C语言.ObjC的特性,且克服了C语言的兼容性问题. Swift发展过程中不仅保留了ObjC很多语法特性,它也借鉴了多种现代化语言的特点,在其中 ...
- 常用的排序算法介绍和在JAVA的实现(二)
一.写随笔的原因:本文接上次的常用的排序算法介绍和在JAVA的实现(一) 二.具体的内容: 3.交换排序 交换排序:通过交换元素之间的位置来实现排序. 交换排序又可细分为:冒泡排序,快速排序 (1)冒 ...
- asyncio动态添加任务
asyncio.run_forever()下动态添加任务 方法一.asyncio.run_coroutine_threadsafe(coroutine, loop) 方法二.asyncio.call_ ...
- 第01章 部署虚拟环境安装Linux系统
在VMware中安装RHEL系统和其它Linux系统一样,注意的是: ……前边一直操作下边的步骤后: 重启系统后将看到系统的初始化界面,单击 LICENSE INFORMATION 选项. 选中 I ...
- dedecms织梦做中英文(多语言)网站步骤详解
用dedecms织梦程序如何做中英文网站,下面是一个详细的图文教程,希望能帮助到大家. 以下是用dedecms织梦程序制作过的一个5国语言网站,下面开始教程. 一.首先在后台建栏目,有三点需要注意 1 ...
- TIOBE 2017 8月编程语言排行榜 后院“硝烟四起”
处于排名榜最前面的几个编程语言的分数长期以来一直都在下降:Java和C在TIOBE榜单中的分数一直比较低.而且几乎所有其他排名前十的语言每年都在下降. 那么哪个什么语言抓住了这个机遇呢?这发生在排行榜 ...
- MonkeyRunner的简介与综合实践
官方介绍: Monkeyrunner工具提供了一个API,用于编写可从Android代码外部控制Android设备或模拟器的程序.使用monkeyrunner,您可以编写一个Python程序来安装An ...
- spring+mybatis 多数据源的配置
方式一: 参见博客https://www.cnblogs.com/AmbitiousMice/p/6027674.html 此种方式每次需要在调用dao的时候设置对应的数据源. 方式二: 直接在myb ...
- 提高docker加载速度
由于国情,我们需要对配置一下docker的下载镜像,提高一下后续的加载速度. 使用vim编辑 /etc/docker/daemon.json, 增加如下内容. { "registry-mir ...