摘要:構(gòu)造器提供了三個構(gòu)造器接口約定,每個集合類應(yīng)該提供兩個標準構(gòu)造器,一個是無參數(shù)的構(gòu)造器上面第一個,另外一個是擁有單個參數(shù)類型為的構(gòu)造器上面第二個。還提供了第三個構(gòu)造器,其接受一個值,用于設(shè)置的初始大小默認大小為。
ArrayList介紹
List 接口的一個實現(xiàn)類,內(nèi)部是用一個數(shù)組存儲元素值,相當于一個可變大小的數(shù)組。
簽名public class ArrayListextends AbstractList implements List , RandomAccess, Cloneable, Serializable
可以看到ArrayList繼承了AbstractList抽象類,它實現(xiàn)了List接口的大多數(shù)方法。如果要實現(xiàn)一個不可變的List,只要繼承這個類并且實現(xiàn)get(int)和size方法。如果要實現(xiàn)可變的List,需要覆蓋set(int, E)。另外,如果List的大小是可變的,還要覆蓋add(int, E)和remove()方法。
構(gòu)造器ArrayList提供了三個構(gòu)造器:
ArrayList() ArrayList(Collection extends E> c) ArrayList(int initialCapacity)
Collection接口約定,每個集合類應(yīng)該提供兩個”標準”構(gòu)造器,一個是無參數(shù)的構(gòu)造器(上面第一個),另外一個是擁有單個參數(shù)(類型為Collettion)的構(gòu)造器(上面第二個)。ArrayList還提供了第三個構(gòu)造器,其接受一個int值,用于設(shè)置ArrayLi的初始大小(默認大小為10)。
相關(guān)方法 trimToSizepublic void trimToSize() { modCount++; int oldCapacity = elementData.length; if (size < oldCapacity) { elementData = Arrays.copyOf(elementData, size); } }
用于把ArrayList的容量縮減到當前實際大小,減少存儲容量。其中的變量modCount由AbstracList繼承而來,記錄List發(fā)生結(jié)構(gòu)化修改(structurally modified)的次數(shù)。elementData中實際存儲了ArrayList的元素,在ArrayList中聲明為:private transient Object[] elementData;變量size是ArrayList的元素數(shù)量,當size < oldCapacity時,調(diào)用Arrays.copyOf方法實現(xiàn)縮減。
indexOf 和 lasIndexOfpublic int indexOf(Object o) { if (o == null) { for (int i = 0; i < size; i++) if (elementData[i]==null) return i; } else { for (int i = 0; i < size; i++) if (o.equals(elementData[i])) return i; } return -1; }
這兩個方法返回指定元素的下標,要區(qū)分參數(shù)是否為null。lastIndexOf和indexOf類似,只不過是從后往前搜索。
ensureCapacitypublic void ensureCapacity(int minCapacity) { if (minCapacity > 0) ensureCapacityInternal(minCapacity); } private void ensureCapacityInternal(int minCapacity) { modCount++; // overflow-conscious code if (minCapacity - elementData.length > 0) grow(minCapacity); } private void grow(int minCapacity) { // overflow-conscious code int oldCapacity = elementData.length; int newCapacity = oldCapacity + (oldCapacity >> 1); if (newCapacity - minCapacity < 0) newCapacity = minCapacity; if (newCapacity - MAX_ARRAY_SIZE > 0) newCapacity = hugeCapacity(minCapacity); // minCapacity is usually close to size, so this is a win: elementData = Arrays.copyOf(elementData, newCapacity); }
這個方法可以確保ArrayList的大小
add 和 addAllpublic void add(int index, E element) { rangeCheckForAdd(index); ensureCapacityInternal(size + 1); // Increments modCount!! System.arraycopy(elementData, index, elementData, index + 1, size - index); elementData[index] = element; size++; }
add(int index, E element)向指定位置添加元素,首先調(diào)用rangeCheckForAdd檢查index是否有效,如果index > size || index < 0將拋出異常。然后確保容量加1,調(diào)用System.arraycopy把從index開始的元素往后移動一個位置。最后把index處的值設(shè)置為添加的元素。還有一個重載的add(E e)方法是直接把元素添加到末尾。
addAll(Collection extends E> c)和addAll(int index, Collection extends E> c)則分別是向末尾和指定位置添加Collection中的所有元素。
public boolean remove(Object o) { if (o == null) { for (int index = 0; index < size; index++) if (elementData[index] == null) { fastRemove(index); return true; } } else { for (int index = 0; index < size; index++) if (o.equals(elementData[index])) { fastRemove(index); return true; } } return false; }
remove(Object o)方法刪除指定的元素。首先是查找元素位置,然后調(diào)用fastRemove(index)刪除,其代碼如下:
private void fastRemove(int index) { modCount++; int numMoved = size - index - 1; if (numMoved > 0) //把index+1往后的元素都往前移動一個位置 System.arraycopy(elementData, index+1, elementData, index, numMoved); elementData[--size] = null; // Let gc do its work }
重載的remove(int index)方法用于刪除指定位置的元素。removeRange(int fromIndex, int toIndex)用于刪除指定位置之間的所有元素。
removeAll(Collection> c)和retainAll(Collection> c)代碼如下:
public boolean removeAll(Collection> c) { Objects.requireNonNull(c); return batchRemove(c, false); } public boolean retainAll(Collection> c) { Objects.requireNonNull(c); return batchRemove(c, true); }
它們都是通過調(diào)用batchRemove方法實現(xiàn)的,其代碼如下:
private boolean batchRemove(Collection> c, boolean complement) { final Object[] elementData = this.elementData; int r = 0, w = 0; boolean modified = false; try { for (; r < size; r++) if (c.contains(elementData[r]) == complement) elementData[w++] = elementData[r]; } finally { // Preserve behavioral compatibility with AbstractCollection, // even if c.contains() throws. if (r != size) { System.arraycopy(elementData, r, elementData, w, size - r); w += size - r; } if (w != size) { // clear to let GC do its work for (int i = w; i < size; i++) elementData[i] = null; modCount += size - w; size = w; modified = true; } } return modified; }
這個方法有兩個參數(shù),第一個是操作的Collection,第二個是一個布爾值,通過設(shè)置為true或false來選擇是removeAll還是retainAll。try里面的語句是把留下來的放在0到w之間,然后在finally中第二個if處理w之后的空間,第一個是在c.contains()拋出異常時執(zhí)行。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/64527.html
摘要:需要注意的是,通過構(gòu)造函數(shù)定義初始量是動態(tài)數(shù)組的實際大小。帶容量的構(gòu)造函數(shù)新建一個容量為的數(shù)組默認構(gòu)造函數(shù),默認為空構(gòu)造一個包含指定元素的第一個構(gòu)造方法使用提供的來初始化數(shù)組的大小。 前言 今天介紹經(jīng)常使用的一個Java集合類——ArrayList(基于JDK1.8.0_121)。ArrayList在工作和日常面試中經(jīng)常被使用或者提到。總的來說,工作中使用ArrayList主要是因為動...
摘要:關(guān)于的具體實現(xiàn),一些基本的都也知道,譬如數(shù)組實現(xiàn),線程不安全等等,但是更加具體的就很少去了解了,例如初始化的長度,擴容等。 前言 在之前的文章中我們提到過ArrayList,ArrayList可以說是每一個學java的人使用最多最熟練的集合了,但是知其然不知其所以然。關(guān)于ArrayList的具體實現(xiàn),一些基本的都也知道,譬如數(shù)組實現(xiàn),線程不安全等等,但是更加具體的就很少去了解了,例如:...
摘要:關(guān)于的具體實現(xiàn),一些基本的都也知道,譬如數(shù)組實現(xiàn),線程不安全等等,但是更加具體的就很少去了解了,例如初始化的長度,擴容等。 前言 在之前的文章中我們提到過ArrayList,ArrayList可以說是每一個學java的人使用最多最熟練的集合了,但是知其然不知其所以然。關(guān)于ArrayList的具體實現(xiàn),一些基本的都也知道,譬如數(shù)組實現(xiàn),線程不安全等等,但是更加具體的就很少去了解了,例如:...
摘要:單線程的迭代過程中刪除集合元素以上代碼會出現(xiàn)如下異常從后往前看第行代碼我們在執(zhí)行代碼行時調(diào)用了這個是調(diào)用返回的對象這個對象的方法如下圖方法首先它會調(diào)用這個方法這個方法很簡單就是比較這兩個值是不是相等不相等就拋出異常如下圖這兩個值為什么會不相 單線程的Iterator迭代過程中刪除集合元素 public class TestIterator { public static voi...
摘要:單線程的迭代過程中刪除集合元素以上代碼會出現(xiàn)如下異常從后往前看第行代碼我們在執(zhí)行代碼行時調(diào)用了這個是調(diào)用返回的對象這個對象的方法如下圖方法首先它會調(diào)用這個方法這個方法很簡單就是比較這兩個值是不是相等不相等就拋出異常如下圖這兩個值為什么會不相 單線程的Iterator迭代過程中刪除集合元素 public class TestIterator { public static voi...
閱讀 1030·2023-04-25 22:27
閱讀 877·2021-11-22 14:56
閱讀 992·2021-11-11 16:54
閱讀 1688·2019-08-30 15:54
閱讀 3509·2019-08-30 13:20
閱讀 1219·2019-08-30 10:55
閱讀 2087·2019-08-26 13:34
閱讀 3287·2019-08-26 11:53