ArrayList 底层以一个transient 线性数组来存储数据,它提供了无参构造方法,和有参构造方法,用户可以通过有参构造方法来初始化长度。如果不传参数,则默认调用无参构造器,数组默认长度为10。

关于数组长度的问题,ArrayList专门提供了一个方法来管理,以避免数组越界。如下:

方法参数为将要达到的长度(newLen),将它与数组实际使用的长度进行比较(factLen),如果newLen>factLen那么将要拓展数组长度。拓展的量为(factLen*3)/2+1(实际上这种情况下,数组的使用长度和开辟长度是相等的),为什么是这种算法?实际上它是在原来的基础上增加50%+1,之所以要加1是因为,执行这个方法的时候通常是需要往数组里面添加数据,数组需要为当前数据保留一个位置,从而保证数组预留位置为原来的1.5倍.

这个方法在何时执行呢,搜索发现,在四个方法执行,分别是add(E),add(index,E),addAll(c),add(index,c),也就只要需要往数组里面添加数据,都需要对数组索引及长度进行判断,以避免数组越界。

而且可以发现,当数组长度重新拓展以后,会把当前数组的数组复制到新长度的数组。这将是非常影响效率的操作。

ArrayList的get()和add()方法都比较简单,首先都需要进行数组越界的考量判断,然后通过索引得到相应的值,值得注意的一点是,add(index ,e)这个方法,做了两个判断,索引小于0和大于当前实际使用值,都是不合法的。比如,初始化后数组长度为10,里面存了5条数据,当前最大使用的索引为4,此时调用add(5,obj),这是会抛出数组越界。这是为ArrarList为了保证分配连续的内存空间。

ArrayList的remove方法,如果传的是一个索引值,首先判断索引是否越界,然后执行删除,并返回删除值。如果传的是一个对象,那么首先会判断对象是否为空,如果为空,遍历数组,找到则执行类似于remove(index)的方法,区别在于没有返回值,如此可见,ArrayList是允许空值的;如果不为空,同上。

既然它可以为空,那么可不可以覆盖掉它,根据前面所说,索引小于0和大于当前实际使用值,都是不合法的,会不会抛异常呢?测试了下,发现不会,新的值占用了原先的位置,但数组实际所使用的长度不变,证明与其说是覆盖,不如说是把原值挤走了。为NULL的值优先级不高。而且,在GC工作的时候可能被回收。如图:

需要注意的是,ArrayList在删除某条数据所做的处理,仅仅只是将该处索引的值设为NULL。这时整个数组的实际使用长度是没有做出改变的,如果刚好逼进临界点,在下次增加时会拓展长度。

只有在clear()方法的时候,才会对此做出处理。

说说两个重要的属性,transient ,modCount,ArrayLIst里保存数据的数组修饰符为transient,表示它不作为序列化的一部分,在进行读写操作的时候需要调用方法。readObject,writeObject,而modCount负责记录ArrayList里操作或者叫做数组改变的次数,在进行读写操作的时候,用modCount进行标记,如果值改变,表明此时另外的线程在对集合进行修改,此时抛出异常ConcurrentModificationException()。

结属说说用得较少的两个方法:

有一个set(index,e)方法,它会用新值覆盖旧值,并返回被覆盖掉的值。

还有一个listIndexOf()方法,这个方法返回相应元素在数组里面最后出现的索引,没有返回-1。遍历,从最后一位开始。

或许你不知道:ArrayList的更多相关文章

  1. 或许你不知道(2):LinkedList

    一,基本的存储结构及数据存取 LinkedList与ArrayList同属List的范畴,ArrayList实现了RandomAccess接口,通过索引随机访问效率较高,而LinkedList提供了直 ...

  2. Java中List接口重要实现类一ArrayList

    1.java.util.ArrayList 集合数据存储的结构是数组结构.元素查找快,而增删就比较慢 所以如果要查询数据,遍历数据,ArrayList是最常用的集合 2.ArrayList是不同步的, ...

  3. Python爬取CSDN博客文章

    0 url :http://blog.csdn.net/youyou1543724847/article/details/52818339Redis一点基础的东西目录 1.基础底层数据结构 2.win ...

  4. Retrofit2.0源码解析

    欢迎访问我的个人博客 ,原文链接:http://wensibo.net/2017/09/05/retrofit/ ,未经允许不得转载! 今天是九月的第四天了,学校也正式开学,趁着大学最后一年的这大好时 ...

  5. 如何深入理解Java泛型

    一.泛型的作用与定义 1.1泛型的作用 使用泛型能写出更加灵活通用的代码泛型的设计主要参照了C++的模板,旨在能让人写出更加通用化,更加灵活的代码.模板/泛型代码,就好像做雕塑时的模板,有了模板,需要 ...

  6. 一次发现underscore源码bug的经历以及对学术界『拿来主义』的思考

    事情是如何发生的 最近干了件事情,发现了 underscore 源码的一个 bug.这件事本身并没有什么可说的,但是过程值得我们深思,记录如下,各位看官仁者见仁智者见智. 平时有浏览园区首页文章的习惯 ...

  7. Java程序员们最常犯的10个错误

    将数组转化为一个列表时,程序员们经常这样做: List<String> list = Arrays.asList(arr); Arrays.asList()会返回一个ArrayList对象 ...

  8. [转]Java程序员们最常犯的10个错误

    1.将数组转化为列表 将数组转化为一个列表时,程序员们经常这样做: List<String> list = Arrays.asList(arr); Arrays.asList()会返回一个 ...

  9. 《FreeSWITCH: VoIP实战》:SIP 模块 - mod_sofia

    SIP 模块是 FreeSWITCH 的主要模块,所以,值得拿出专门一章来讲解. 在前几章时里,你肯定见过几次 sofia 这个词,只是或许还不知道是什么意思.是这样的,Sofia-SIP 是由诺基亚 ...

随机推荐

  1. pthread_cond_wait避免线程空转

    多线程对同一块区域进行操作时,需要定义如下两种类型的变量: pthread_mutex_t xxx; pthread_cond_t yyy; pthread_mutex_t类型的变量,即锁,对公共区域 ...

  2. HTTP状态

    HTTP状态码 当浏览者访问一个网页时,浏览者的浏览器会向网页所在服务器发出请求.当浏览器接收并显示网页前,此网页所在的服务器会返回一个包含HTTP状态码的信息头(server header)用以响应 ...

  3. CentOS 6.4安装Kangle面板

    kangle web server一键安装包是一个用Linux Shell编写的可以为CentOS 6 VPS(VDS)或独立主机安装kangle web server(kangle,easypane ...

  4. iOS中UIKit——UIFont得到iOS设备上的系统字体

    for (NSString *fontFamily  in [UIFont familyNames]) { NSLog(@"字体家族是:%@",fontFamily); for(N ...

  5. Stop PeopleCode Processing with Error

    A blunt, but useful method for debugging PeopleCode is to use the inbuilt "Error" function ...

  6. 如何在linux中配置PHP环境

    yum -y install httpd php mysql mysql-server php-mysql//安装mysql扩展yum -y install mysql-connector-odbc( ...

  7. echarts简单使用案例

    先上效果图:

  8. 新浪SAE URLRewrite(伪静态、重定向)详解

    SAE全称Sina App Engine,真是一个好东西,他有很多优秀的特性,简单来说SAE就是一个简单高效的分布式Web服务开发.运行平台.支持现在常用的 PHP+Mysql 环境,在开发中难免会碰 ...

  9. Yii 增删改查 测试记录

    亲们, 我是yii小白 不要笑话我奥.今天白天写一个管理模块涉及到 yii ar 下的  curd 操作,做 update 操作时纠结了好久,今天晚上花点时间学习, 下面写下我的测试记录 代码如下: ...

  10. js实现touch移动触屏滑动事件

    在开始描述touch事件之前,需要先描述一下多触式系统中特有的touch对象(android和iOS乃至nokia最新的meego系统都模拟了类 似的对象).这个对象封装一次屏幕触摸,一般来自于手指. ...