基於jdk_1.8.0html
關於List,主要是有序的可重複的數據結構。jdk主要實現類有ArrayList(底層使用數組)、LinkedList(底層使用雙向鏈表)
ArrayList:java
(一)繼承關係圖數組
(二)源碼解析安全
(1)關鍵字段 數據結構
1 /** 2 * 默認初始容量 3 */ 4 private static final int DEFAULT_CAPACITY = 10; 5 6 /** 7 * 用於空實例的共享空數組實例 8 */ 9 private static final Object[] EMPTY_ELEMENTDATA = {}; 10 11 /** 12 * 用於默認大小的空實例的共享空數組實例 13 */ 14 private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; 15 16 /** 17 *存儲ArrayList元素的數組緩衝區 18 */ 19 transient Object[] elementData; //非私有以簡化嵌套類訪問 20 21 /** 22 * ArrayList的大小,也是下一個元素插入的位置 23 */ 24 private int size; 25 26 /** 27 * 要分配的數組的最大大小 28 * 從hugeCapacity方法能夠看到若是要分配的大小 > MAX_ARRAY_SIZE ,會返回Integer.MAX_VALUE 29 */ 30 private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8; // 關於-8 ,網上說是數組元數據信息要佔8個字節 字節和大小有哪門子關係,這個不夠讓人信服
(2)構造方法 多線程
1 /** 2 * elementData指向共享的默認空數組實例,將在添加第一個元素時擴展到DEFAULT_CAPACITY 3 */ 4 public ArrayList() { 5 //默認的空數組實例,主要是爲了區別 new ArrayList(0) 6 this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; 7 } 8 9 /** 10 * 構造一個initialCapacity大小的數組 11 */ 12 public ArrayList(int initialCapacity) { 13 if (initialCapacity > 0) { 14 this.elementData = new Object[initialCapacity]; 15 } else if (initialCapacity == 0) { 16 //空數組實例 17 this.elementData = EMPTY_ELEMENTDATA; 18 } else { 19 throw new IllegalArgumentException("Illegal Capacity: "+ 20 initialCapacity); 21 } 22 } 23 24 /** 25 * 構造一個包含指定集合的元素的列表,這些元素按集合的迭代器返回的順序排列 26 */ 27 public ArrayList(Collection<? extends E> c) { 28 //詳見toArray實例方法 29 elementData = c.toArray(); 30 //集合不爲空 31 if ((size = elementData.length) != 0) { 32 //雖然elementData 是object[]類型的,可是它指向的類型不必定是Object[] 33 //形成的緣由多是Arrays.asList詳情請戳https://www.cnblogs.com/liqing-weikeyuan/p/7922306.html 34 if (elementData.getClass() != Object[].class) 35 elementData = Arrays.copyOf(elementData, size, Object[].class); 36 } else { 37 this.elementData = EMPTY_ELEMENTDATA; 38 } 39 }
(3)經常使用方法dom
a. public boolean add(E e) ide
1 public boolean add(E e) { 2 // 爲了確保數組不越界,因此容許的數組最小容量爲size + 1 3 ensureCapacityInternal(size + 1); // Increments modCount!! 4 elementData[size++] = e; 5 return true; 6 } 7 8 private void ensureCapacityInternal(int minCapacity) { 9 // 因爲calculateCapacity是靜態方法,沒法訪問實例elementData變量 10 ensureExplicitCapacity(calculateCapacity(elementData, minCapacity)); 11 } 12 13 /** 14 * 爲啥要設計成靜態方法,想不通 15 */ 16 private static int calculateCapacity(Object[] elementData, int minCapacity) { 17 // 如果經過默認構造方法構造,則返回DEFAULT_CAPACITY 18 if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) { 19 return Math.max(DEFAULT_CAPACITY, minCapacity); 20 } 21 return minCapacity; 22 } 23 24 private void ensureExplicitCapacity(int minCapacity) { 25 modCount++; 26 27 // overflow-conscious code 28 // 若數組不足以放下,則進行擴充 29 if (minCapacity - elementData.length > 0) 30 grow(minCapacity); 31 } 32 33 /** 34 * 新的數組容量先按以前的1.5倍進行擴充 35 * 若擴充後的大小還不足以放下,則使用minCapacity 36 * 若minCapacity > MAX_ARRAY_SIZE,最大容量Integer.MAX_VALUE,不然內存溢出異常 37 * @param minCapacity 38 */ 39 private void grow(int minCapacity) { 40 // overflow-conscious code 41 int oldCapacity = elementData.length; 42 int newCapacity = oldCapacity + (oldCapacity >> 1); 43 if (newCapacity - minCapacity < 0) 44 newCapacity = minCapacity; 45 if (newCapacity - MAX_ARRAY_SIZE > 0) 46 newCapacity = hugeCapacity(minCapacity); 47 // minCapacity is usually close to size, so this is a win: 48 elementData = Arrays.copyOf(elementData, newCapacity); 49 } 50 51 private static int hugeCapacity(int minCapacity) { 52 // 容量超過Integer.MAX_VALUE 53 if (minCapacity < 0) // overflow 54 throw new OutOfMemoryError(); 55 return (minCapacity > MAX_ARRAY_SIZE) ? 56 Integer.MAX_VALUE : 57 MAX_ARRAY_SIZE; 58 }
b. public void add(int index, E element)性能
1 /** 2 * 向指定位置添加元素,不經常使用,性能也很差 3 * @param index 4 * @param element 5 */ 6 public void add(int index, E element) { 7 rangeCheckForAdd(index); 8 9 ensureCapacityInternal(size + 1); // Increments modCount!! 10 // [index, size -1] 向後移動一位 11 System.arraycopy(elementData, index, elementData, index + 1, 12 size - index); 13 elementData[index] = element; 14 size++; 15 } 16 17 /** 18 * A version of rangeCheck used by add and addAll. 19 */ 20 private void rangeCheckForAdd(int index) { 21 if (index > size || index < 0) 22 throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); 23 } 24 25 /** 26 * 詳見 a. public boolean add(E e) 再也不展開 27 * @param minCapacity 28 */ 29 private void ensureCapacityInternal(int minCapacity) { 30 ensureExplicitCapacity(calculateCapacity(elementData, minCapacity)); 31 }
c. public boolean addAll(Collection<? extends E> c)ui
1 public boolean addAll(Collection<? extends E> c) { 2 Object[] a = c.toArray(); 3 int numNew = a.length; 4 ensureCapacityInternal(size + numNew); // 詳見a. add 5 System.arraycopy(a, 0, elementData, size, numNew); 6 size += numNew; 7 return numNew != 0; //??? 暫時沒看懂 8 }
d. public boolean addAll(int index, Collection<? extends E> c)
1 /** 2 * 向指定位置添加集合元素 3 * @param index 4 * @param c 5 * @return 6 */ 7 public boolean addAll(int index, Collection<? extends E> c) { 8 rangeCheckForAdd(index); 9 10 Object[] a = c.toArray(); 11 int numNew = a.length; 12 ensureCapacityInternal(size + numNew); // Increments modCount 13 14 // 須要移動元素個數 15 int numMoved = size - index; 16 if (numMoved > 0) 17 System.arraycopy(elementData, index, elementData, index + numNew, 18 numMoved); //從index開始複製numMoved個元素到index + numNew (即[index, index + numMoved -1] 複製到[index + numNew, index + numNew + numMoved - 1]) 19 20 System.arraycopy(a, 0, elementData, index, numNew); 21 size += numNew; 22 return numNew != 0; //??? 沒看懂 23 }
e. public E remove(int index)
1 /** 2 * 刪除指定位置的元素 3 * @param index 4 * @return 被刪除的元素 5 * @throws java.lang.ArrayIndexOutOfBoundsException 6 */ 7 public E remove(int index) { 8 rangeCheck(index); 9 10 modCount++; 11 E oldValue = elementData(index); 12 13 int numMoved = size - index - 1; 14 if (numMoved > 0) 15 System.arraycopy(elementData, index+1, elementData, index, 16 numMoved); //把[index + 1, size -1]元素 複製到 [index, size -2] 17 elementData[--size] = null; // clear to let GC do its work 18 19 return oldValue; 20 } 21 22 private void rangeCheck(int index) { 23 if (index >= size) 24 throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); 25 } 26 27 private String outOfBoundsMsg(int index) { 28 return "Index: "+index+", Size: "+size; 29 } 30 31 /** 32 * @throws java.lang.ArrayIndexOutOfBoundsException 33 */ 34 E elementData(int index) { 35 return (E) elementData[index]; 36 }
f. public boolean remove(Object o)
1 /** 2 * 刪除第一個值爲o的元素 3 * @param o 4 * @return 5 */ 6 public boolean remove(Object o) { 7 if (o == null) { 8 for (int index = 0; index < size; index++) 9 if (elementData[index] == null) { 10 fastRemove(index); 11 return true; 12 } 13 } else { 14 for (int index = 0; index < size; index++) 15 if (o.equals(elementData[index])) { 16 fastRemove(index); 17 return true; 18 } 19 } 20 return false; 21 } 22 23 private void fastRemove(int index) { 24 modCount++; 25 int numMoved = size - index - 1; 26 if (numMoved > 0) 27 System.arraycopy(elementData, index+1, elementData, index, 28 numMoved); 29 elementData[--size] = null; // clear to let GC do its work 30 }
g. public boolean removeAll(Collection<?> c)
1 public boolean removeAll(Collection<?> c) { 2 Objects.requireNonNull(c); 3 return batchRemove(c, false); 4 } 5 6 // Objects.class 7 public static <T> T requireNonNull(T obj) { 8 if (obj == null) 9 throw new NullPointerException(); 10 return obj; 11 } 12 13 private boolean batchRemove(Collection<?> c, boolean complement) { 14 final Object[] elementData = this.elementData; 15 int r = 0, w = 0; 16 boolean modified = false; 17 try { 18 for (; r < size; r++) 19 if (c.contains(elementData[r]) == complement) 20 elementData[w++] = elementData[r]; // 把沒有包含的統一移動到數組的一端 21 } finally { 22 // Preserve behavioral compatibility with AbstractCollection, 23 // even if c.contains() throws. 24 if (r != size) { 25 // 前r個元素已通過濾完畢,這裏只是簡單的把[r, size-1]的元素複製移動到[w, w + size -r -1] 26 System.arraycopy(elementData, r, 27 elementData, w, 28 size - r); 29 w += size - r; 30 } 31 if (w != size) { 32 // clear to let GC do its work 33 for (int i = w; i < size; i++) 34 elementData[i] = null; 35 modCount += size - w; 36 size = w; 37 modified = true; 38 } 39 } 40 return modified; 41 }
h. public E set(int index, E element)
1 public E set(int index, E element) { 2 rangeCheck(index); 3 4 E oldValue = elementData(index); //強制轉換成E 5 elementData[index] = element; 6 return oldValue; 7 }
i. public E get(int index)
1 public E get(int index) { 2 rangeCheck(index); 3 4 return elementData(index); // 強制轉化成E 5 }
j. public int indexOf(Object o)
1 /** 2 * 獲取指定元素的下標 3 * @param o 4 * @return 5 */ 6 public int indexOf(Object o) { 7 if (o == null) { 8 for (int i = 0; i < size; i++) 9 if (elementData[i]==null) 10 return i; 11 } else { 12 for (int i = 0; i < size; i++) 13 if (o.equals(elementData[i])) 14 return i; 15 } 16 return -1; 17 }
k. public int lastIndexOf(Object o)
1 public int lastIndexOf(Object o) { 2 if (o == null) { 3 for (int i = size-1; i >= 0; i--) 4 if (elementData[i]==null) 5 return i; 6 } else { 7 for (int i = size-1; i >= 0; i--) 8 if (o.equals(elementData[i])) 9 return i; 10 } 11 return -1; 12 }
l. public Iterator<E> iterator()
1 public Iterator<E> iterator() { 2 return new Itr(); 3 } 4 5 /** 6 * An optimized version of AbstractList.Itr 7 */ 8 private class Itr implements Iterator<E> { 9 int cursor; // index of next element to return 10 int lastRet = -1; // index of last element returned; -1 if no such 11 int expectedModCount = modCount; 12 13 Itr() {} 14 15 public boolean hasNext() { 16 return cursor != size; 17 } 18 19 @SuppressWarnings("unchecked") 20 public E next() { 21 checkForComodification(); 22 int i = cursor; 23 if (i >= size) //數組內實際存儲的元素個數 24 throw new NoSuchElementException(); 25 Object[] elementData = ArrayList.this.elementData; 26 if (i >= elementData.length) 27 throw new ConcurrentModificationException(); 28 cursor = i + 1; 29 return (E) elementData[lastRet = i]; 30 } 31 32 public void remove() { 33 if (lastRet < 0) //沒有首先調用next(), 直接remove會拋IllegalStateException 34 throw new IllegalStateException(); 35 checkForComodification(); 36 37 try { 38 ArrayList.this.remove(lastRet); //詳見e. public E remove(int index) 39 cursor = lastRet; 40 lastRet = -1; 41 expectedModCount = modCount; 42 } catch (IndexOutOfBoundsException ex) { //該異常什麼狀況下會被觸發??? 43 throw new ConcurrentModificationException(); 44 } 45 } 46 47 @Override 48 @SuppressWarnings("unchecked") 49 public void forEachRemaining(Consumer<? super E> consumer) { 50 Objects.requireNonNull(consumer); 51 final int size = ArrayList.this.size; 52 int i = cursor; 53 if (i >= size) { 54 return; 55 } 56 final Object[] elementData = ArrayList.this.elementData; 57 if (i >= elementData.length) { 58 throw new ConcurrentModificationException(); 59 } 60 while (i != size && modCount == expectedModCount) { 61 consumer.accept((E) elementData[i++]); 62 } 63 // update once at end of iteration to reduce heap write traffic 64 cursor = i; 65 lastRet = i - 1; 66 checkForComodification(); 67 } 68 69 final void checkForComodification() { 70 if (modCount != expectedModCount) 71 throw new ConcurrentModificationException(); 72 } 73 }
m. public ListIterator<E> listIterator()
1 public ListIterator<E> listIterator() { 2 return new ListItr(0); 3 } 4 5 /** 6 * An optimized version of AbstractList.ListItr 7 */ 8 private class ListItr extends Itr implements ListIterator<E> { 9 // 初始化遊標位置,默認0 10 ListItr(int index) { 11 super(); 12 cursor = index; 13 } 14 15 // 只有第一個元素沒有前繼節點 16 public boolean hasPrevious() { 17 return cursor != 0; 18 } 19 20 public int nextIndex() { 21 return cursor; 22 } 23 24 public int previousIndex() { 25 return cursor - 1; 26 } 27 28 @SuppressWarnings("unchecked") 29 public E previous() { 30 checkForComodification(); 31 int i = cursor - 1; 32 if (i < 0) 33 throw new NoSuchElementException(); 34 Object[] elementData = ArrayList.this.elementData; 35 if (i >= elementData.length) 36 throw new ConcurrentModificationException(); 37 cursor = i; 38 return (E) elementData[lastRet = i]; 39 } 40 41 public void set(E e) { 42 if (lastRet < 0) 43 throw new IllegalStateException(); 44 checkForComodification(); 45 46 try { 47 ArrayList.this.set(lastRet, e); // 詳見 h. public E set(int index, E element) 48 } catch (IndexOutOfBoundsException ex) { 49 throw new ConcurrentModificationException(); 50 } 51 } 52 53 /** 54 * 因爲底層調用的是add(int index, E element), 還要有一次數組拷貝,性能不如 add(E element) 55 * @param e 56 */ 57 public void add(E e) { 58 checkForComodification(); 59 60 try { 61 int i = cursor; 62 ArrayList.this.add(i, e); //詳見b. public void add(int index, E element) 63 cursor = i + 1; 64 lastRet = -1; 65 expectedModCount = modCount; 66 } catch (IndexOutOfBoundsException ex) { 67 throw new ConcurrentModificationException(); 68 } 69 } 70 }
n. public ListIterator<E> listIterator(int index)
1 /** 2 * 指定遊標 3 * @param index 4 * @return 5 */ 6 public ListIterator<E> listIterator(int index) { 7 if (index < 0 || index > size) 8 throw new IndexOutOfBoundsException("Index: "+index); 9 return new ListItr(index); // 詳情請看 j. public ListIterator<E> listIterator() 10 }
o. public void forEach(Consumer<? super E> action)
1 @Override 2 public void forEach(Consumer<? super E> action) { 3 Objects.requireNonNull(action); 4 final int expectedModCount = modCount; 5 @SuppressWarnings("unchecked") 6 final E[] elementData = (E[]) this.elementData; 7 final int size = this.size; 8 for (int i=0; modCount == expectedModCount && i < size; i++) { 9 action.accept(elementData[i]); //使用見下面解析 10 } 11 if (modCount != expectedModCount) { 12 throw new ConcurrentModificationException(); 13 } 14 } 15 16 // Consumer.class 17 @FunctionalInterface // 使用該註解就代表可使用lambda,只要接口只有一個抽象方法就能夠用lambda 18 public interface Consumer<T> { 19 20 void accept(T t); 21 22 /** 23 * 能夠看到andThen方法返回的是一個Consumer匿名類對象c 24 * 程序運行中若是不調用c.accept(),則什麼也不會發生 25 * @param after 26 * @return 27 */ 28 default Consumer<T> andThen(Consumer<? super T> after) { 29 Objects.requireNonNull(after); 30 return (T t) -> { accept(t); after.accept(t); }; //重寫了accept方法 31 } 32 } 33 34 // TestConsumer.class 35 public void testAccept(String[] args) { 36 // jdk8 以前寫法 37 Consumer<String> c = new Consumer<String>() { 38 @Override 39 public void accept(String s) { 40 System.out.println(s.toLowerCase()); 41 } 42 }; 43 44 // jdk8 lambda寫法 45 Consumer<String> c = (s) -> System.out.println(s.toLowerCase()); 46 c.accept("HELLO world!"); // hello world! 47 } 48 49 public void testAndthen(String[] args) { 50 /** 51 * 經過這個例子能夠看出首先調用了a.accept(),而後調用了b.accept() 52 */ 53 Consumer<String> a = (x) -> System.out.print(x.toLowerCase()); 54 Consumer<String> b = (x) -> { 55 System.out.println("...andThen..." + x); 56 }; 57 Consumer<String> c = a.andThen(b); 58 c.accept("HELLO world!"); // hello world!...andThen...HELLO world! 59 }
p. public void sort(Comparator<? super E> c) //TODO 先不展開
1 @Override 2 @SuppressWarnings("unchecked") 3 public void sort(Comparator<? super E> c) { 4 final int expectedModCount = modCount; 5 Arrays.sort((E[]) elementData, 0, size, c); 6 if (modCount != expectedModCount) { 7 throw new ConcurrentModificationException(); 8 } 9 modCount++; 10 }
q. public List<E> subList(int fromIndex, int toIndex)
1 /** 2 * 注意,該subList並無new一個新的object[], 對subList對象的修改,都會做用到原大對象上 3 */ 4 public List<E> subList(int fromIndex, int toIndex) { 5 subListRangeCheck(fromIndex, toIndex, size); 6 return new SubList(this, 0, fromIndex, toIndex); 7 } 8 9 static void subListRangeCheck(int fromIndex, int toIndex, int size) { 10 if (fromIndex < 0) 11 throw new IndexOutOfBoundsException("fromIndex = " + fromIndex); 12 if (toIndex > size) 13 throw new IndexOutOfBoundsException("toIndex = " + toIndex); 14 if (fromIndex > toIndex) 15 throw new IllegalArgumentException("fromIndex(" + fromIndex + 16 ") > toIndex(" + toIndex + ")"); 17 } 18 19 /** 20 * subList 並無再new一個object[],而是複用 21 */ 22 private class SubList extends AbstractList<E> implements RandomAccess { 23 private final AbstractList<E> parent; 24 private final int parentOffset; 25 private final int offset; 26 int size; 27 28 SubList(AbstractList<E> parent, 29 int offset, int fromIndex, int toIndex) { 30 this.parent = parent; 31 this.parentOffset = fromIndex; 32 this.offset = offset + fromIndex; 33 this.size = toIndex - fromIndex; 34 this.modCount = ArrayList.this.modCount; 35 } 36 37 public E set(int index, E e) { 38 rangeCheck(index); 39 checkForComodification(); 40 E oldValue = ArrayList.this.elementData(offset + index); 41 ArrayList.this.elementData[offset + index] = e; 42 return oldValue; 43 } 44 45 public E get(int index) { 46 rangeCheck(index); 47 checkForComodification(); 48 return ArrayList.this.elementData(offset + index); 49 } 50 51 public int size() { 52 checkForComodification(); 53 return this.size; 54 } 55 56 public void add(int index, E e) { 57 rangeCheckForAdd(index); 58 checkForComodification(); 59 parent.add(parentOffset + index, e); 60 this.modCount = parent.modCount; 61 this.size++; 62 } 63 64 public E remove(int index) { 65 rangeCheck(index); 66 checkForComodification(); 67 E result = parent.remove(parentOffset + index); 68 this.modCount = parent.modCount; 69 this.size--; 70 return result; 71 } 72 73 protected void removeRange(int fromIndex, int toIndex) { 74 checkForComodification(); 75 parent.removeRange(parentOffset + fromIndex, 76 parentOffset + toIndex); 77 this.modCount = parent.modCount; 78 this.size -= toIndex - fromIndex; 79 } 80 81 public boolean addAll(Collection<? extends E> c) { 82 return addAll(this.size, c); 83 } 84 85 public boolean addAll(int index, Collection<? extends E> c) { 86 rangeCheckForAdd(index); 87 int cSize = c.size(); 88 if (cSize==0) 89 return false; 90 91 checkForComodification(); 92 parent.addAll(parentOffset + index, c); 93 this.modCount = parent.modCount; 94 this.size += cSize; 95 return true; 96 } 97 98 public Iterator<E> iterator() { 99 return listIterator(); 100 } 101 102 public ListIterator<E> listIterator(final int index) { 103 checkForComodification(); 104 rangeCheckForAdd(index); 105 final int offset = this.offset; 106 107 return new ListIterator<E>() { 108 int cursor = index; 109 int lastRet = -1; 110 int expectedModCount = ArrayList.this.modCount; 111 112 public boolean hasNext() { 113 return cursor != SubList.this.size; 114 } 115 116 @SuppressWarnings("unchecked") 117 public E next() { 118 checkForComodification(); 119 int i = cursor; 120 if (i >= SubList.this.size) 121 throw new NoSuchElementException(); 122 Object[] elementData = ArrayList.this.elementData; 123 if (offset + i >= elementData.length) 124 throw new ConcurrentModificationException(); 125 cursor = i + 1; 126 return (E) elementData[offset + (lastRet = i)]; 127 } 128 129 public boolean hasPrevious() { 130 return cursor != 0; 131 } 132 133 @SuppressWarnings("unchecked") 134 public E previous() { 135 checkForComodification(); 136 int i = cursor - 1; 137 if (i < 0) 138 throw new NoSuchElementException(); 139 Object[] elementData = ArrayList.this.elementData; 140 if (offset + i >= elementData.length) 141 throw new ConcurrentModificationException(); 142 cursor = i; 143 return (E) elementData[offset + (lastRet = i)]; 144 } 145 146 @SuppressWarnings("unchecked") 147 public void forEachRemaining(Consumer<? super E> consumer) { 148 Objects.requireNonNull(consumer); 149 final int size = SubList.this.size; 150 int i = cursor; 151 if (i >= size) { 152 return; 153 } 154 final Object[] elementData = ArrayList.this.elementData; 155 if (offset + i >= elementData.length) { 156 throw new ConcurrentModificationException(); 157 } 158 while (i != size && modCount == expectedModCount) { 159 consumer.accept((E) elementData[offset + (i++)]); 160 } 161 // update once at end of iteration to reduce heap write traffic 162 lastRet = cursor = i; 163 checkForComodification(); 164 } 165 166 public int nextIndex() { 167 return cursor; 168 } 169 170 public int previousIndex() { 171 return cursor - 1; 172 } 173 174 public void remove() { 175 if (lastRet < 0) 176 throw new IllegalStateException(); 177 checkForComodification(); 178 179 try { 180 SubList.this.remove(lastRet); 181 cursor = lastRet; 182 lastRet = -1; 183 expectedModCount = ArrayList.this.modCount; 184 } catch (IndexOutOfBoundsException ex) { 185 throw new ConcurrentModificationException(); 186 } 187 } 188 189 public void set(E e) { 190 if (lastRet < 0) 191 throw new IllegalStateException(); 192 checkForComodification(); 193 194 try { 195 ArrayList.this.set(offset + lastRet, e); 196 } catch (IndexOutOfBoundsException ex) { 197 throw new ConcurrentModificationException(); 198 } 199 } 200 201 public void add(E e) { 202 checkForComodification(); 203 204 try { 205 int i = cursor; 206 SubList.this.add(i, e); 207 cursor = i + 1; 208 lastRet = -1; 209 expectedModCount = ArrayList.this.modCount; 210 } catch (IndexOutOfBoundsException ex) { 211 throw new ConcurrentModificationException(); 212 } 213 } 214 215 final void checkForComodification() { 216 if (expectedModCount != ArrayList.this.modCount) 217 throw new ConcurrentModificationException(); 218 } 219 }; 220 } 221 222 public List<E> subList(int fromIndex, int toIndex) { 223 subListRangeCheck(fromIndex, toIndex, size); 224 return new SubList(this, offset, fromIndex, toIndex); 225 } 226 227 private void rangeCheck(int index) { 228 if (index < 0 || index >= this.size) 229 throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); 230 } 231 232 private void rangeCheckForAdd(int index) { 233 if (index < 0 || index > this.size) 234 throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); 235 } 236 237 private String outOfBoundsMsg(int index) { 238 return "Index: "+index+", Size: "+this.size; 239 } 240 241 private void checkForComodification() { 242 if (ArrayList.this.modCount != this.modCount) 243 throw new ConcurrentModificationException(); 244 } 245 246 public Spliterator<E> spliterator() { 247 checkForComodification(); 248 return new ArrayListSpliterator<E>(ArrayList.this, offset, 249 offset + this.size, this.modCount); 250 } 251 }
r. public <T> T[] toArray(T[] a)
1 /** 2 * 推薦使用 toArray(T[] a) 3 */ 4 public Object[] toArray() { 5 return Arrays.copyOf(elementData, size); 6 } 7 8 /** 9 * 若是a.length不足以放下全部的元素,則新建一個size大小的數組拷貝返回 10 * 若是a.length足夠放下,遍歷的時候當心空指針 11 * 推薦a.length == size new T[list.size()] 12 */ 13 public <T> T[] toArray(T[] a) { 14 if (a.length < size) // 傳過來的數組大小不足以放下所有元素 15 // Make a new array of a's runtime type, but my contents: 16 return (T[]) Arrays.copyOf(elementData, size, a.getClass()); 17 System.arraycopy(elementData, 0, a, 0, size); 18 if (a.length > size) 19 a[size] = null; // ???沒懂用意 20 return a; 21 }
s. 其它簡單方法
1 /** 2 * 返回當前列表元素個數 3 * @return 4 */ 5 public int size() { 6 return size; 7 } 8 9 /** 10 * 當前列表是否爲空 11 * @return 12 */ 13 public boolean isEmpty() { 14 return size == 0; 15 } 16 17 /** 18 * 當前列表是否包含元素o 19 * @return 20 */ 21 public boolean contains(Object o) { 22 return indexOf(o) >= 0; //詳情請看 j. public int indexOf(Object o) 23 } 24 25 /** 26 * 將elementData中未使用的空間 釋放掉 27 */ 28 public void trimToSize() { 29 modCount++; 30 if (size < elementData.length) { 31 elementData = (size == 0) 32 ? EMPTY_ELEMENTDATA 33 : Arrays.copyOf(elementData, size); 34 } 35 } 36 37 /** 38 * 清空全部引用 39 */ 40 public void clear() { 41 modCount++; 42 43 // clear to let GC do its work 44 for (int i = 0; i < size; i++) 45 elementData[i] = null; 46 47 size = 0; 48 }
t. 序列化相關方法
1 private void readObject(java.io.ObjectInputStream s) 2 throws java.io.IOException, ClassNotFoundException { 3 elementData = EMPTY_ELEMENTDATA; 4 5 // Read in size, and any hidden stuff 6 s.defaultReadObject(); 7 8 // Read in capacity 9 s.readInt(); // ignored 10 11 if (size > 0) { 12 // be like clone(), allocate array based upon size not capacity 13 int capacity = calculateCapacity(elementData, size); 14 SharedSecrets.getJavaOISAccess().checkArray(s, Object[].class, capacity); 15 ensureCapacityInternal(size); 16 17 Object[] a = elementData; 18 // Read in all elements in the proper order. 19 for (int i=0; i<size; i++) { 20 a[i] = s.readObject(); 21 } 22 } 23 } 24 25 private void writeObject(java.io.ObjectOutputStream s) 26 throws java.io.IOException{ 27 // Write out element count, and any hidden stuff 28 int expectedModCount = modCount; 29 s.defaultWriteObject(); 30 31 // Write out size as capacity for behavioural compatibility with clone() 32 s.writeInt(size); 33 34 // Write out all elements in the proper order. 35 for (int i=0; i<size; i++) { 36 s.writeObject(elementData[i]); 37 } 38 39 if (modCount != expectedModCount) { 40 throw new ConcurrentModificationException(); 41 } 42 }
寫在最後:
使用ArrayList進行存儲的元素,務必要重寫Object的equals方法。像IndexOf、lastIndexOf、remove等方法底層都是調用元素的equals方法進行比較;
經過get、set方法能夠看出,ArrayList隨機讀寫指定位置的元素速度很是快,可是remove、add等方法可能還須要有移動操做,故速度較慢;
經過remove indexOf 等方法能夠看出ArrayList能夠存儲null值;
ArrayList.size 區間範圍[0, Integer.MAX_VALUE], 超過Integer.MAX_VALUE,拋OutOfMemoryError。實際應用中還須要具體調整JVM內存大小;
經過代碼能夠看出,ArrayList任何方法均沒有鎖,故ArrayList是非線程安全的,若要在多線程環境下使用,要麼調用者在外部同步要麼參考Collections.synchronizedList(new ArrayList());
ListIterator加強了Iterator一些功能,添加了add、set、hasPrevious、previousIndex、previous、nextIndex方法,調用remove方法時,務必首先調用next或者previous;
subList 應該注意,對subList對象的任何修改都會做用到原對象上;
toArray,返回的是一個全新的對象,對其修改不會影響原list對象。
Q1:for-each與iterator使用場景
A1:若是一個集合或者數組須要更新元素(修改|刪除),那麼須要使用iterator或者普通for循環;
若是對於集合List使用iterator,建議優先選擇使用listIterator;
若是隻遍歷集合元素,建議優先使用forEach或者支持並行的stream(jdk8以上),不然使用for-each。