设定初始化容量new一个list

List<String> list = new ArrayList<>(10);
list.add(“s”);
  • 进入构造方法
//属性elementData,ArrayList就是一个数组
transient Object[] elementData; public ArrayList(int initialCapacity) {
//没啥特殊的,容量大于0就直接定义个initialCapacity长度的数组,赋值给initialCapacity
if (initialCapacity > 0) {
this.elementData = new Object[initialCapacity];
} else if (initialCapacity == 0) {
//参数为0,直接给elementData赋值空的数组
this.elementData = EMPTY_ELEMENTDATA;
} else {
//小于0,抛异常
throw new IllegalArgumentException("Illegal Capacity: "+
initialCapacity);
}
}
  • 进入 add()---> 参数为泛型类型,也就是elementData数组中元素类型
public boolean add(E e) {
//计数器,开始为0
modCount++;
add(e, elementData, size);
return true;
}
  • 直接进入add(e, elementData, size)

    • elementData是一个空的数组,长度为10
    • size现在应该是0 没有被赋值
- 进入三个参数的add()

private void add(E e, Object[] elementData, int s) {

//可以知道,s指的就是数组的索引

if (s == elementData.length)

//如果索引刚好等于数组长度,那么需要扩容!

elementData = grow();

//否则,直接在数组s索引的位置,赋值e

elementData[s] = e;

//索引 + 1

size = s + 1;

- 到这里,添加目测应该是结束了,进入`grow()`看看写了什么东西

private Object[] grow() {

return grow(size + 1);

}

- 继续进入`grow(size + 1)`,注意:传入参数为初始化容量 +1 ,在本例中为 11

private Object[] grow(int minCapacity) {

return elementData = Arrays.copyOf(elementData,

newCapacity(minCapacity));

}



- 不难发现,其实扩容操作就是针对elementData 属性
- 将此数组容量经过`newCapacity`(),形成一个扩容后新的数组,赋值给原数组!
- 那么扩容了多少呢?
- 进入`newCapacity(minCapacity)`

private int newCapacity(int minCapacity) {

// overflow-conscious code

//oldCapacity赋值为原数组长度 = 10
int oldCapacity = elementData.length;
//oldCapacity + oldCapacity /2 = 1.5 * oldCapacity
//newCapacity = 15
int newCapacity = oldCapacity + (oldCapacity >> 1);
//出现意外了!newCapacity 明明是1.5倍的newCapacity
//为什么 <= newCapacity 了,如下2种情况!
if (newCapacity - minCapacity <= 0) {
if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA)
//情况1,数组为空,即newCapacity = 0
//返回常量 DEFAULT_CAPACITY 10
return Math.max(DEFAULT_CAPACITY, minCapacity);
if (minCapacity < 0) // overflow
//入参 < 0 ,抛异常
throw new OutOfMemoryError();
return minCapacity;
}
//判断 newCapacity 是不是特别大(大到只比最大的Integer小8)
//是的话,就进入hugeCapacity() ,不做讨论
//正常的话,直接返回一个newCapacity
return (newCapacity - MAX_ARRAY_SIZE <= 0)
? newCapacity
: hugeCapacity(minCapacity);

}


- 那么问题来了,为什么扩容之前,要size+1,在本例中,入参为11
`int newCapacity = oldCapacity + (oldCapacity >> 1);`
- 采用如上方式计算新的容量,在于 >> 1
- 也就是转成二进制之后右移1
- 对于1,3,5,7等奇数,二进制最后一位是 1 ,右移后末尾的1直接没了
- 所以要 +1,使其变成偶数,在乘1.5
- 本例中,10+1变成11,右移,得到5
- 如果是9,变成10后,右移,得到5
- 综上,若参数为`m`,实际扩容的容量数值 = m=奇数 ? m+1 * 1.5 : m*1.5!

ArrayList相关问题的更多相关文章

  1. Java中ArrayList相关的5道面试题

    本文参考了 <关于ArrayList的5道面试题 > 1.ArrayList的大小是如何自动增加的? 这个问题我想曾经debug过并且查看过arraylist源码的人都有印象,它的过程是: ...

  2. JDK源码分析(3)之 ArrayList 相关

    ArrayList的源码其实比较简单,所以我并没有跟着源码对照翻译,文本只是抽取了一些我觉得有意思或一些有疑惑的地方分析的. 一.成员变量 private static final int DEFAU ...

  3. ArrayList

    各种原因,前两年做C语言去了,现在重新做JAVA, 感觉自己基础很不扎实,要好好学习啦, 先从简单的开始~ 以下内容基于jdk1.7.0_79源码: 什么是ArrayList 可以简单的认为是一个动态 ...

  4. (转载)关于ArrayList的5道面试题

    我以面试官的身份参加过很多Java的面试,以下是五个比较有技巧的问题,我发现有些初级到中级的Java研发人员在这些问题上没有完全弄明白,似懂非懂.所以我写了一篇相关的文章,帮助初级Java研发人员弄清 ...

  5. JDK1.8源码阅读系列之一:ArrayList

    本篇随笔主要描述的是我阅读 ArrayList 源码期间的对于 ArrayList 的一些实现上的个人理解,有不对的地方,请指出- 先来看一下 ArrayList 的继承图: 由图可以看出,Array ...

  6. 关于ArrayList的5道面试题

    我以面试官的身份参加过很多Java的面试,以下是五个比较有技巧的问题,我发现有些初级到中级的Java研发人员在这些问题上没有完全弄明白,似懂非懂.所以我写了一篇相关的文章,帮助初级Java研发人员弄清 ...

  7. 再说Java集合,subList之于ArrayList

    上一章说了很多ArrayList相关的内容,但还有一块儿内容没说到,那就是subList方法.先看一段代码 public static void testSubList() { List<Str ...

  8. Java8集合框架——ArrayList源码分析

    java.util.ArrayList 以下为主要介绍要点,从 Java 8 出发: 一.ArrayList的特点概述 二.ArrayList的内部实现:从内部属性和构造函数说起 三.ArrayLis ...

  9. 代码审查:从 ArrayList 说线程安全

    本文从代码审查过程中发现的一个 ArrayList 相关的「线程安全」问题出发,来剖析和理解线程安全. 案例分析 前两天在代码 Review 的过程中,看到有小伙伴用了类似以下的写法: List< ...

随机推荐

  1. pta 习题集5-18 打印学生选课清单

    假设全校有最多40000名学生和最多2500门课程.现给出每门课的选课学生名单,要求输出每个前来查询的学生的选课清单. 输入格式: 输入的第一行是两个正整数:N(≤≤40000),为前来查询课表的学生 ...

  2. docx4j基本操作

    最近需要用docx4j来对docx进行一些操作,用到的技术是docx4j,这个技术在国内其实用的不是很多,看了一些博主的文章,有些感悟,做了一些总结,如果有疑问或错误之处欢迎交流. 创建包: Word ...

  3. Oracle管理监控之监控表空间使用率脚本

    SELECT D.TABLESPACE_NAME,       SPACE "SUM_SPACE(M)",       BLOCKS SUM_BLOCKS,       SPACE ...

  4. describe neural networks as a series of computational steps via a directed graph.

    https://www.microsoft.com/en-us/research/product/cognitive-toolkit/ https://github.com/microsoft/cnt ...

  5. Introduction to Structured Data json的2种形式 JAVA解析JSON数据 - JsonArray JsonObject

    https://developers.google.com/search/docs/guides/intro-structured-data Structured data refers to kin ...

  6. cocos2d首印象

    一. 创建工程 从 2.1.4 版本开始,官方就不再为 VS 提供模板了,逐步在各平台采用统一的 Python 脚本创建跨平台工程. 要创建工程,我们需要先从命令行进入 tools/project-c ...

  7. PAT 1135 Is It A Red-Black Tree[难]

    1135 Is It A Red-Black Tree (30 分) There is a kind of balanced binary search tree named red-black tr ...

  8. python 面向对象编程学习总结

    面向对象是个抽象的东西,概念比较多,下面会一一介绍. 一.类和实例 类(Class)和实例(Instance)是面向对象最重要的概念. 类是指抽象出的模板.实例则是根据类创建出来的具体的“对象”,每个 ...

  9. Django进阶 (二)

    规范 确立规范的好处: 代码可读性高 方便代码的定位极其查找 为以后代码扩容带来便利 场景: 在多个APP的场景下,单个app的URL函数功能较多的时候,我们可以通过以下方法来解决. 把Views写成 ...

  10. 关于Serializable的一个形象的例子

    一.知识预备 根据使用Serializable的使用场景,可以发现所涉及的场景都是跨进程的,就是要做的事情不是在一个java进程中完成的,我们都知道java进程是基于jvm跑起来的,而每一个被创建出来 ...