java基础 之 list源码分析(ArrayList)

ArrayList:
继承关系分析:
  1. public class ArrayList<E> extends AbstractList<E>
  2. implements List<E>, RandomAccess, Cloneable, java.io.Serializable

我们可以知道:

  1. 继承了AbstractList

  2. 实现了List接口

  3. 实现了RandomAccess,这里举例说明下这个接口的作用,我们看一段代码:

    Collections类中的binarySearch方法:

    1. public static <T>
    2. int binarySearch(List<? extends Comparable<? super T>> list, T key) {
    3. // 实现了RandomAccess接口或者集合长度小于5000
    4. if (list instanceof RandomAccess || list.size()<BINARYSEARCH_THRESHOLD)
    5. return Collections.indexedBinarySearch(list, key);
    6. else
    7. return Collections.iteratorBinarySearch(list, key);
    8. }

    可以看到,如果实现了RandomAccess接口或者集合长度小于5000,会调用indexedBinarySearch,否则调用iteratorBinarySearch。

    我们来看下这个两个方法的区别:

    1. int indexedBinarySearch(List<? extends Comparable<? super T>> list, T key) {
    2. int low = 0;
    3. int high = list.size()-1;
    4. while (low <= high) {
    5. int mid = (low + high) >>> 1;
    6. // 在这里采用的是直接通过下标获取指定元素的方式
    7. Comparable<? super T> midVal = list.get(mid);
    8. int cmp = midVal.compareTo(key);
    9. if (cmp < 0)
    10. low = mid + 1;
    11. else if (cmp > 0)
    12. high = mid - 1;
    13. else
    14. return mid; // key found
    15. }
    16. return -(low + 1); // key not found
    17. }
    1. private static <T>
    2. int iteratorBinarySearch(List<? extends Comparable<? super T>> list, T key)
    3. {
    4. int low = 0;
    5. int high = list.size()-1;
    6. // 这里采用的是迭代器的方式
    7. ListIterator<? extends Comparable<? super T>> i = list.listIterator();
    8. while (low <= high) {
    9. int mid = (low + high) >>> 1;
    10. Comparable<? super T> midVal = get(i, mid);
    11. int cmp = midVal.compareTo(key);
    12. if (cmp < 0)
    13. low = mid + 1;
    14. else if (cmp > 0)
    15. high = mid - 1;
    16. else
    17. return mid; // key found
    18. }
    19. return -(low + 1); // key not found
    20. }

    我们查看源码可以发现,ArrayList实现了RandomAccess接口,所以ArrayList会通过索引直接访问,而LinkedList没有实现RandomAccess接口,会采用迭代器的方式访问,这主要是跟他们的数据结构有关,ArrayList底层是数组,LinkedList底层是链表,具体的原因就不详细说了,留给读者自行思考,有问题可以留言一起讨论。

  4. 实现了Cloneable,代表可以被克隆

  5. 实现了Serializable,代表了可以被序列化

属性分析:
  1. /**
  2. * 初始容量
  3. */
  4. private static final int DEFAULT_CAPACITY = 10;
  5. /**
  6. * 当采用public ArrayList(Collection<? extends E> c)这种构造函数时
  7. * 若传入的集合对象大小为0的话,此时用这个空数组作为这个ArrayList的元素
  8. */
  9. private static final Object[] EMPTY_ELEMENTDATA = {};
  10. /**
  11. *当调用空参构造时,用这个空数组作为ArrayList的元素,虽然都是空数组,但是空参构造时
  12. *会采用默认容量DEFAULT_CAPACITY = 10,是为了区分开是哪种方式构造的这个ArrayList
  13. */
  14. private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
  15. /**
  16. *实际存储了ArrayList的元素的集合
  17. *这里可以思考一个问题:为什么要使用transient关键字修饰?
  18. *我会在后文中详细解释
  19. */
  20. transient Object[] elementData; // non-private to simplify nested class access
  21. /**
  22. * 存储了的实际元素的数量,这里主要要跟容量区分开
  23. *我们可以这样理解,size是实际存的数量,而CAPACITY(容量)是打算存的数量
  24. */
  25. private int size;
  26. /**
  27. *可分配的最大数组长度,实际上最大可分配到Integer.MAX_VALUE,后面我们结合源码分析
  28. *减8是因为有些虚拟机需要存储一些头信息,稍后我们会分析下为什么要减8
  29. */
  30. private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;
  31. /**
  32. *从父类AbstractList中继承而来的属性,记录了集合被修改的次数,主要为了实现快速失败机制
  33. *后面在方法分析中在详细解释
  34. */
  35. protected transient int modCount = 0;

在上面的属性分析,我们遗留了一个问题,即elementData为什么要使用transient关键字修饰?

现在来详细解释下:

​ 我们知道transient用来表示一个域不是该对象序列化的一部分,当一个对象被序列化的时候,transient修饰的变量的值是不包括在序列化的表示中的。但是ArrayList又是可序列化的类,elementData是ArrayList具体存放元素的成员,用transient来修饰elementData,岂不是反序列化后的ArrayList丢失了原先的元素?这里就要说到我们后面要说到的两个方法

  • private void writeObject(java.io.ObjectOutputStream s)

  • private void readObject(java.io.ObjectInputStream s)

    ​ 对这两个方法的分析,我们放到后文中去,这里先说下,之所以这样设计主要是因为elementData是一个缓存数组,它通常会预留一些容量,等容量不足时再扩充容量,那么有些空间可能就没有实际存储元素,采用上诉的方式来实现序列化时,就可以保证只序列化实际存储的那些元素,而不是整个数组,从而节省空间和时间。

构造函数分析:
  1. /**
  2. * 构造一个指定了初始容量的ArrayList,参数为负数的话,抛出IllegalArgumentException异常
  3. *
  4. * @param initialCapacity 初始容量
  5. *
  6. */
  7. public ArrayList(int initialCapacity) {
  8. if (initialCapacity > 0) {
  9. this.elementData = new Object[initialCapacity];
  10. } else if (initialCapacity == 0) {
  11. this.elementData = EMPTY_ELEMENTDATA;
  12. } else {
  13. throw new IllegalArgumentException("Illegal Capacity: "+
  14. initialCapacity);
  15. }
  16. }
  17. /**
  18. * 空参构造,当第一次调用add方法时,将会采用默认值DEFAULT_CAPACITY=10作为初始容量
  19. */
  20. public ArrayList() {
  21. this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
  22. }
  23. /**
  24. *构造一个包含了指定集合元素的ArrayList,如果集合元素为空,则此时ArrayList的容量也是0,
  25. *不会采用默认容量
  26. */
  27. public ArrayList(Collection<? extends E> c) {
  28. elementData = c.toArray();
  29. if ((size = elementData.length) != 0) {
  30. // c.toArray might (incorrectly) not return Object[] (see 6260652)
  31. if (elementData.getClass() != Object[].class)
  32. elementData = Arrays.copyOf(elementData, size, Object[].class);
  33. } else {
  34. // replace with empty array.
  35. this.elementData = EMPTY_ELEMENTDATA;
  36. }
  37. }
方法分析:
  • add(E e)
  1. /**
  2. *添加指定的元素到集合末尾
  3. */
  4. public boolean add(E e) {
  5. // 确保容量,继续跟踪这个方法
  6. ensureCapacityInternal(size + 1);
  7. elementData[size++] = e;
  8. return true;
  9. }
  10. private void ensureCapacityInternal(int minCapacity) {
  11. // 在属性分析的适合我们已经说过了,如果是DEFAULTCAPACITY_EMPTY_ELEMENTDATA,代表是通过
  12. // 空参构造创建的ArrayList,这个时候
  13. if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
  14. minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
  15. }
  16. // 继续跟踪这个方法
  17. ensureExplicitCapacity(minCapacity);
  18. }
  19. private void ensureExplicitCapacity(int minCapacity) {
  20. modCount++;
  21. // 要求的最小容量如果大于了现有的数组长度就进行扩容
  22. if (minCapacity - elementData.length > 0)
  23. // 继续跟踪这个方法
  24. grow(minCapacity);
  25. }
  26. private void grow(int minCapacity) {
  27. // 这里主要是做了一些溢出的考虑
  28. int oldCapacity = elementData.length;
  29. // 如果相加的和大于了int的最大值的话,这里就会得到一个负数,右移相当于模2
  30. int newCapacity = oldCapacity + (oldCapacity >> 1);
  31. // 如果是负数的话,相减肯定小于0
  32. if (newCapacity - minCapacity < 0)
  33. // 小于0的话,就将minCapacity(要求的最小容量,就是原有size+1)的值赋值给newCapacity(扩容后的 // 容量)
  34. newCapacity = minCapacity;
  35. // 如果扩容后的容量大于了 MAX_ARRAY_SIZE
  36. if (newCapacity - MAX_ARRAY_SIZE > 0)
  37. // 没有OOM发生的话,就干脆将newCapacity置为int的最大值
  38. newCapacity = hugeCapacity(minCapacity);
  39. // minCapacity is usually close to size, so this is a win:
  40. elementData = Arrays.copyOf(elementData, newCapacity);
  41. }

分析了add方法后,我们就可以对ArrayList的扩容机制有了一个很全面的了解:

  • 第一次调用add后,如果是通过空参构造的话,默认会给一个10的初始容量

  • 添加元素时,会判断要求的最小容量(size+1)是否超出了现有的数组长度,如果超出了要进行扩容

  • 扩容时,会在原有容量的基础上进行1.5倍的扩容

  • 如果扩容后的长度超出了int的最大值,就用size+1作为本次扩容后的容量

  • 如果size+1大于了MAX_ARRAY_SIZE,就干脆用int的最大值作为容量

    从上面也可以看出,如果add方法在添加的时候,不需要进行扩容的话,添加元素也是很快的,只需要将size+1上的指针指向指定元素就行了,如果涉及到扩容的话,性能就不高了,因为要移动一部分数组元素,并且添加元素的位置越靠前,移动的元素越多

    • add(int index, E element)
    1. // 在分析了add方法后,对于这个重载方法就不用花太多时间了
    2. public void add(int index, E element) {
    3. rangeCheckForAdd(index);
    4. ensureCapacityInternal(size + 1); // Increments modCount!!
    5. // 这里相当于将数组从index位置开始到数据末尾的所有元素往后移动一位,然后将移动后数组上的index位置置为新添加的element元素
    6. System.arraycopy(elementData, index, elementData, index + 1,
    7. size - index);
    8. elementData[index] = element;
    9. size++;

    }

    • clear()
    1. // 这个方法非常简单,就是把所有的元素置为null,并且集合修改次数加1
    2. public void clear() {
    3. modCount++;
    4. // clear to let GC do its work
    5. for (int i = 0; i < size; i++)
    6. elementData[i] = null;
    7. size = 0;
    8. }
    • ensureCapacity(int minCapacity)
    1. // 在集合完成初始化后,调用进行手动扩容
    2. public void ensureCapacity(int minCapacity) {
    3. // 首先判断是通过什么方式初始化的,然后给一个初始容量
    4. int minExpand = (elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA)
    5. ? 0
    6. : DEFAULT_CAPACITY;
    7. // 如果大于默认容量,进行扩容
    8. if (minCapacity > minExpand) {
    9. ensureExplicitCapacity(minCapacity);
    10. }
    11. }
    • E get(int index)

      1. public E get(int index) {
      2. // 检查是否角标越界
      3. rangeCheck(index);
      4. // 直接从数组中获取index位置上的元素,效率很高
      5. return elementData(index);
      6. }
    • Iterator iterator()

      1. // 返回内部的迭代器
      2. public Iterator&lt;E&gt; iterator() {
      3. return new Itr();
      4. }

      我们接下来探究下迭代器的源码:

      1. private class Itr implements Iterator<E> {
      2. int cursor; // index of next element to return
      3. int lastRet = -1; // index of last element returned; -1 if no such
      4. int expectedModCount = modCount;
      5. public boolean hasNext() {
      6. return cursor != size;
      7. }
      8. @SuppressWarnings("unchecked")
      9. public E next() {
      10. checkForComodification();
      11. int i = cursor;
      12. if (i >= size)
      13. throw new NoSuchElementException();
      14. Object[] elementData = ArrayList.this.elementData;
      15. if (i >= elementData.length)
      16. throw new ConcurrentModificationException();
      17. cursor = i + 1;
      18. return (E) elementData[lastRet = i];
      19. }
      20. public void remove() {
      21. if (lastRet < 0)
      22. throw new IllegalStateException();
      23. checkForComodification();
      24. try {
      25. ArrayList.this.remove(lastRet);
      26. cursor = lastRet;
      27. lastRet = -1;
      28. expectedModCount = modCount;
      29. } catch (IndexOutOfBoundsException ex) {
      30. throw new ConcurrentModificationException();
      31. }
      32. }
      33. @Override
      34. @SuppressWarnings("unchecked")
      35. public void forEachRemaining(Consumer<? super E> consumer) {
      36. Objects.requireNonNull(consumer);
      37. final int size = ArrayList.this.size;
      38. int i = cursor;
      39. if (i >= size) {
      40. return;
      41. }
      42. final Object[] elementData = ArrayList.this.elementData;
      43. if (i >= elementData.length) {
      44. throw new ConcurrentModificationException();
      45. }
      46. while (i != size && modCount == expectedModCount) {
      47. consumer.accept((E) elementData[i++]);
      48. }
      49. // update once at end of iteration to reduce heap write traffic
      50. cursor = i;
      51. lastRet = i - 1;
      52. checkForComodification();
      53. }
      54. final void checkForComodification() {
      55. if (modCount != expectedModCount)
      56. throw new ConcurrentModificationException();
      57. }
      58. }
    • ListIterator listIterator()

      1. // 返回另外一个迭代器
      2. public ListIterator<E> listIterator() {
      3. return new ListItr(0);
      4. }

      再看看这个迭代器的实现有什么区别?

      1. // 继承了Itr
      2. private class ListItr extends Itr implements ListIterator<E> {
      3. ListItr(int index) {
      4. super();
      5. cursor = index;
      6. }
      7. // 可以向前遍历
      8. public boolean hasPrevious() {
      9. return cursor != 0;
      10. }
      11. public int nextIndex() {
      12. return cursor;
      13. }
      14. public int previousIndex() {
      15. return cursor - 1;
      16. }
      17. @SuppressWarnings("unchecked")
      18. public E previous() {
      19. checkForComodification();
      20. int i = cursor - 1;
      21. if (i < 0)
      22. throw new NoSuchElementException();
      23. Object[] elementData = ArrayList.this.elementData;
      24. if (i >= elementData.length)
      25. throw new ConcurrentModificationException();
      26. cursor = i;
      27. return (E) elementData[lastRet = i];
      28. }
      29. // 新增了set方法
      30. public void set(E e) {
      31. if (lastRet < 0)
      32. throw new IllegalStateException();
      33. checkForComodification();
      34. try {
      35. ArrayList.this.set(lastRet, e);
      36. } catch (IndexOutOfBoundsException ex) {
      37. throw new ConcurrentModificationException();
      38. }
      39. }
      40. // 新增了add方法
      41. public void add(E e) {
      42. checkForComodification();
      43. try {
      44. int i = cursor;
      45. ArrayList.this.add(i, e);
      46. cursor = i + 1;
      47. lastRet = -1;
      48. expectedModCount = modCount;
      49. } catch (IndexOutOfBoundsException ex) {
      50. throw new ConcurrentModificationException();
      51. }
      52. }
      53. }
  1. **我们可以发现以上两个区别**:
  2. 1. listIterator允许向前遍历
  3. 2. listIterator允许在遍历的过程中添加元素

java读源码 之 list源码分析(ArrayList)---JDK1.8的更多相关文章

  1. Java并发指南10:Java 读写锁 ReentrantReadWriteLock 源码分析

    Java 读写锁 ReentrantReadWriteLock 源码分析 转自:https://www.javadoop.com/post/reentrant-read-write-lock#toc5 ...

  2. Java读源码之ReentrantLock

    前言 ReentrantLock 可重入锁,应该是除了 synchronized 关键字外用的最多的线程同步手段了,虽然JVM维护者疯狂优化 synchronized 使其已经拥有了很好的性能.但 R ...

  3. Java读源码之ReentrantLock(2)

    前言 本文是 ReentrantLock 源码的第二篇,第一篇主要介绍了公平锁非公平锁正常的加锁解锁流程,虽然表达能力有限不知道有没有讲清楚,本着不太监的原则,本文填补下第一篇中挖的坑. Java读源 ...

  4. Java读源码之CountDownLatch

    前言 相信大家都挺熟悉 CountDownLatch 的,顾名思义就是一个栅栏,其主要作用是多线程环境下,让多个线程在栅栏门口等待,所有线程到齐后,栅栏打开程序继续执行. 案例 用一个最简单的案例引出 ...

  5. 源码分析— java读写锁ReentrantReadWriteLock

    前言 今天看Jraft的时候发现了很多地方都用到了读写锁,所以心血来潮想要分析以下读写锁是怎么实现的. 先上一个doc里面的例子: class CachedData { Object data; vo ...

  6. 【转载】深度解读 java 线程池设计思想及源码实现

    总览 开篇来一些废话.下图是 java 线程池几个相关类的继承结构: 先简单说说这个继承结构,Executor 位于最顶层,也是最简单的,就一个 execute(Runnable runnable) ...

  7. Java并发指南12:深度解读 java 线程池设计思想及源码实现

    ​深度解读 java 线程池设计思想及源码实现 转自 https://javadoop.com/2017/09/05/java-thread-pool/hmsr=toutiao.io&utm_ ...

  8. 一篇文章带您读懂List集合(源码分析)

    今天要分享的Java集合是List,主要是针对它的常见实现类ArrayList进行讲解 内容目录 什么是List核心方法源码剖析1.文档注释2.构造方法3.add()3.remove()如何提升Arr ...

  9. 转:微信开发之使用java获取签名signature(贴源码,附工程)

    微信开发之使用java获取签名signature(贴源码,附工程) 标签: 微信signature获取签名 2015-12-29 22:15 6954人阅读 评论(3) 收藏 举报  分类: 微信开发 ...

  10. java基础进阶一:String源码和String常量池

    作者:NiceCui 本文谢绝转载,如需转载需征得作者本人同意,谢谢. 本文链接:http://www.cnblogs.com/NiceCui/p/8046564.html 邮箱:moyi@moyib ...

随机推荐

  1. 【python实现卷积神经网络】批量归一化层实现

    代码来源:https://github.com/eriklindernoren/ML-From-Scratch 卷积神经网络中卷积层Conv2D(带stride.padding)的具体实现:https ...

  2. 1 - Apache HttpClient 简单使用

    Apache HttpClient 是Apache 开源的实现Http协议的java开源库. HttpClien 是客户端的HTTP通信实现库,实现HTTP GET 和POST请求,获取响应内容. A ...

  3. ELK(日志审计系统)

    ELk简介及工作流程 ELK即(Elasticsearch + Logstash + Kibana) 下载安装包 系统环境:Contos7.0 Java环境:Portal(这是历史下载地址,我的是 j ...

  4. 美化你的终端利器Iterm2

    Iterm2是特别好用的一款终端,支持自定义字体和高亮,让日常开发,充满愉悦. 安装iterm2(mac版) brew tap caskroom/cask brew cask install iter ...

  5. vue axios post请求下载文件,后台springmvc完整代码

     注意请求时要设置responseType,不加会中文乱码,被这个坑困扰了大半天... axios post请求:     download(index,row){         var ts =  ...

  6. 通过dockerfile制作镜像

    Dockerfile是一个用于构建Docker镜像的文本文件,其中包含了创建Docker镜像的全部指令.就是将我们安装环境的每个步骤使用指令的形式存放在一个文件中,最后生成一个需要的环境. Docke ...

  7. [PHP][thinkphp5] 学习一:增删改查

    <?php namespace app\index\controller; use think\Controller; use think\Db; class Test extends Cont ...

  8. python之excel表操作

    python对excel表操作主要用到三个库,xlrd,xlwt,xlutils,分别用于excel表读,写,修改操作,以下将一个简单介绍 一.生成excel表:xlwt类 新建excel表,并写入数 ...

  9. 二叉树中两节点的最近公共父节点(360的c++一面问题)

    面试官的问题:写一个函数  TreeNode* Find(TreeNode* root, TreeNode* p, TreeNode* q) ,返回二叉树中p和q的最近公共父节点. 本人反应:当时有点 ...

  10. filter和interceptor的区别

    前言 最近在面试的时候,被问到了这个问题,觉得答得不是很好,在此进行整理和记录,供自己学习,也希望能帮助到大家. 什么是Filter 在java的javax.servlet下有一个接口Filter.任 ...