/** * Shared empty array instance used for empty instances. */ // 一个空数组,主要用于带参数构造函数初始话和读取序列化对象等(方便使用) privatestaticfinal Object[] EMPTY_ELEMENTDATA = {};
/** * Shared empty array instance used for default sized empty instances. We * distinguish this from EMPTY_ELEMENTDATA to know how much to inflate when * first element is added. * DEFAULTCAPACITY_EMPTY_ELEMENTDATA 和EMPTY_ELEMENTDATA 的区别: * 仅仅是为了区别用户带参为0的构造和默认构造的惰性初始模式对象。 * 当用户带参为0的构造,第一次add时,数组容量grow到1; * 当用户使用默认构造时,第一次add时,容量直接grow到DEFAULT_CAPACITY(10); */ privatestaticfinal Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
/** * The array buffer into which the elements of the ArrayList are stored. * The capacity of the ArrayList is the length of this array buffer. Any * empty ArrayList with elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA * will be expanded to DEFAULT_CAPACITY when the first element is added. */ // 当前数据对象存放地方,当前对象不参与序列化 // 这个关键字最主要的作用就是当序列化时,被transient修饰的内容将不会被序列化 // ArraryList另外实现了序列化与反序列化 transient Object[] elementData; // non-private to simplify nested class access
/** * The size of the ArrayList (the number of elements it contains). * * @serial */ // 当前数组数组中的元素 privateint size; /** * The maximum size of array to allocate. * Some VMs reserve some header words in an array. * Attempts to allocate larger arrays may result in * OutOfMemoryError: Requested array size exceeds VM limit */ // 数组最大可分配容量 // 存储的最大容量也取决于运行Java代码的平台内存和JVM的堆比例 // 8个byte是用来存储元数据,用来描述这个数组的长度 privatestaticfinalintMAX_ARRAY_SIZE= Integer.MAX_VALUE - 8;
/** * Constructs an empty list with an initial capacity of ten. */ // 无参(默认)构造函数 publicArrayList() { // 只有这个地方会引用DEFAULTCAPACITY_EMPTY_ELEMENTDATA // 使用这个数组是在添加第一个元素的时候会扩容到默认大小10 this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; }
/** * Constructs an empty list with the specified initial capacity. * * @param initialCapacity the initial capacity of the list * @throws IllegalArgumentException if the specified initial capacity * is negative */ // 当用户带参为0的构造,第一次add时,数组容量grow到1; publicArrayList(int initialCapacity) { if (initialCapacity > 0) { // 如果传入的初始容量大于0,就新建一个数组存储元素 this.elementData = newObject[initialCapacity]; } elseif (initialCapacity == 0) { // 使用EMPTY_ELEMENTDATA,在其他的多个地方可能会引用EMPTY_ELEMENTDATA this.elementData = EMPTY_ELEMENTDATA; } else { thrownewIllegalArgumentException("Illegal Capacity: "+ initialCapacity); } }
/** * Constructs a list containing the elements of the specified * collection, in the order they are returned by the collection's * iterator. * * @param c the collection whose elements are to be placed into this list * @throws NullPointerException if the specified collection is null */ publicArrayList(Collection<? extends E> c) { // 集合转数组给elementData(toArray浅拷贝?) elementData = c.toArray(); if ((size = elementData.length) != 0) { // c.toArray might (incorrectly) not return Object[] (see 6260652) // 若c.toArray()返回的数组类型不是Object[],则利用Arrays.copyOf(); 来构造一个大小为size的Object[]数组 // 此时elementData是指向传入集合的内存,还需要创建新的内存区域深拷贝给elementData if (elementData.getClass() != Object[].class) elementData = Arrays.copyOf(elementData, size, Object[].class); } else { // replace with empty array. this.elementData = EMPTY_ELEMENTDATA; } }
// 删除指定位置的元素 public E remove(int index) { // 检查index范围 rangeCheck(index); modCount++; // 需要删除的元素 EoldValue= elementData(index); // 如果index不是最后一位,则将index之后的元素往前挪一位 intnumMoved= size - index - 1; if (numMoved > 0) System.arraycopy(elementData, index+1, elementData, index, numMoved); // 将最后一个元素删除,帮助GC elementData[--size] = null; // clear to let GC do its work // 返回旧值 return oldValue; }
// 删除指定元素值的元素,时间复杂度为O(n) publicbooleanremove(Object o) { if (o == null) { // 遍历整个数组,找到元素第一次出现的位置,并将其快速删除 for (intindex=0; index < size; index++) // 如果要删除的元素为null,则以null进行比较,使用 “==” if (elementData[index] == null) { fastRemove(index); returntrue; } } else { // 遍历整个数组,找到元素第一次出现的位置,并将其快速删除 for (intindex=0; index < size; index++) // 如果要删除的元素不为null,则进行比较,使用 “equals()” 方法 if (o.equals(elementData[index])) { fastRemove(index); returntrue; } } returnfalse; }
privatevoidfastRemove(int index) { modCount++; // 如果index不是最后一位,则将index之后的元素往前挪一位 intnumMoved= size - index - 1; if (numMoved > 0) System.arraycopy(elementData, index+1, elementData, index, numMoved); elementData[--size] = null; // clear to let GC do its work }
// 序列化 privatevoidwriteObject(java.io.ObjectOutputStream s) throws java.io.IOException{ // Write out element count, and any hidden stuff // 防止序列化期间数组元素有修改 intexpectedModCount= modCount; // 写出非transient非static属性(会写出size属性) s.defaultWriteObject();
// Write out size as capacity for behavioural compatibility with clone() // 序列化数组包含元素数量,为了向后兼容 // 两次将size写入流 s.writeInt(size);
// Write out all elements in the proper order // 按照顺序写入,只写入到数组包含元素的结尾,并不会把数组的所有容量区域全部写入 for (int i=0; i<size; i++) { s.writeObject(elementData[i]); } // 判断是否触发Fast-Fail if (modCount != expectedModCount) { thrownewConcurrentModificationException(); } }
// Read in size, and any hidden stuff // 将流中的的非静态(non-static)和非瞬态(non-transient)字段读取到当前类 // 包含 size s.defaultReadObject();
// Read in capacity // 读入元素个数,没什么用,只是因为写出的时候写了size属性,读的时候也要按顺序来读 s.readInt(); // ignored
if (size > 0) { // be like clone(), allocate array based upon size not capacity // 根据size计算容量 intcapacity= calculateCapacity(elementData, size); // 用于调用另一个包中的实现专用方法,而不使用反射。TODO SharedSecrets.getJavaOISAccess().checkArray(s, Object[].class, capacity); // 检查是否需要扩容 ensureCapacityInternal(size);
Object[] a = elementData; // Read in all elements in the proper order. for (int i=0; i<size; i++) { // 依次读取元素到数组中 a[i] = s.readObject(); } } }