设定初始化容量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. 在线预览文档(支持word、excel、ppt、pdf)+在线预览文档html版(转)

    1.首先上网搜索一下有什么解决方案 (1).将文档转换为html,只支持支持office文档 (2).将文档转换为flash,实现类似百度文库的效果,除支持office文档外还支持pdf (1) a. ...

  2. Tomcat部署静态网站

    公司架构:公司架构有5套,主机都是阿里云的ecs,基本上都是SLB做前端负载均衡,后端Tomcat,后接RDS数据库. 业务需求:需要将公司现有网站指向一个二级域名,建立一个新的静态网站,将域名指向现 ...

  3. Python3、Unicode、UTF-8、编码

    text = u'你好,今天天气不错' text print(text) text = '\u4f60\u597d\uff0c\u4eca\u5929\u5929\u6c14\u4e0d\u9519' ...

  4. Optimal Milking---poj2112(多重匹配+Floyd+二分)

    题目链接:http://poj.org/problem?id=2112 题意:K个挤奶器(编号1~K),每个挤奶器每天最多供M头奶牛.共有C头奶牛(编号K+1~K+C).挤奶器和奶牛间有不同长度的路. ...

  5. find the safest road---hdu1596(最短路模板求最大概率)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1596 求给定的任意两点之间的最大安全概率,概率之间是相乘的关系,所以注意初始化即可 #include& ...

  6. Python 面向对象 类 __str__

    class dog(object): def __init__(self,name): self.name = name d = dog('mike') print(d) # <__main__ ...

  7. Python isdigit() isalnum()

    Python isdigit() 方法检测字符串是否只由数字组成. 返回值 如果字符串只包含数字则返回 True 否则返回 False. >>> choice = input(&qu ...

  8. POJ2992:Divisors(求N!因子的个数,乘性函数,分解n!的质因子(算是找规律))

    题目链接:http://poj.org/problem?id=2992 题目要求:Your task in this problem is to determine the number of div ...

  9. 【Lua】模块与包

    定义:         从用户观点来看,一个模块就是一个程序库,可以通过require来加载(require用于使用模块,module用于创建模块),然后便得到了一个全局变量,表示一个table,这个 ...

  10. source insight 4.0的基本使用方法(转)

    源:source insight 4.0的基本使用方法 source insight 4设置