摘要:實現了接口,所以包含的基本方法新增,刪除,插入等都實現了。線程安全問題在多線程情況下有線程進行修改時,是線程不安全的。線程安全性問題,取決于如何應用。反映的是當前數組中存了多少數據,而則反映的是內部數組的容量。
什么是ArrayList
ArrayList 是一個可擴容數組Resizable-array,它實現了List接口的所有方法。
ArrayList 需要知道的10點內容 1.ArrayList 內部是通過數組實現從對ArrayList的簡單描述中我們可以得出幾點
ArrayList 是數組,但不同于一般數組,可擴容,而一般數組容量固定。
ArrayList 實現了List接口,所以List包含的基本方法(新增,刪除,插入等)ArrayList都實現了。
ArrayList 內部實現是通過一個對象數組2.ArrayList 添加操作(add) 的執行時間是固定的
transient Object[] elementData
ArrayList add() 方法實質上是實現了在數組的末尾添加一個元素 elementData[size++] = el 所以執行的時間復雜度是固定的O(1),添加n個元素為O(n)3.除了 add 方法外其他操作如: 插入、刪除 操作時間是線性的。
因為本質上是對數組進行操作,當在數組中插入、刪除數據時,需要先對數組進行移位操作,插入時需要將插入點以后的數據全部后移,而刪除操作則需要將刪除節點后的數據全部前移,操作時間復雜度為O(n)。4.線程安全問題
ArrayList 在多線程情況下有線程進行修改時,是線程不安全的。線程安全性問題,取決于如何應用。List list = Collections.synchronizedList(new ArrayList(...))可以獲取線程安全的ArrayList5.關于擴容
ArrayList 有其自動擴容機制也可以在預知要處理數據大小時手動擴容
6.關于ArrayList在方法中傳遞的問題。1.自動擴容
ArrayList 在新增元素時(add ,addAll操作)會先檢測當前內部數組elementData[]的容量是否足夠添加新元素,如果不足則擴容,ArrayList最大容量為Integer.MAX_VALUE,自動擴容調用的流程如下
public boolean add(E e) { // 自動擴容,并記錄元素修改數量 ,元素修改數量主要是用于并發修改錯誤 ensureCapacityInternal(size + 1); elementData[size++] = e; return true; } private void ensureCapacityInternal(int minCapacity) { ensureExplicitCapacity(calculateCapacity(elementData, minCapacity)); } // 返回一個數組大小的值 minCapacity 和默認的 DEFAULT_CAPACITY 中較大的那個 private static int calculateCapacity(Object[] elementData, int minCapacity) { // 如果是空數組 通過new ArrayList()創建 if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) { // 默認大小是DEFAULT_CAPACITY = 10 return Math.max(DEFAULT_CAPACITY, minCapacity); } return minCapacity; }實際擴充容量的片段
private void ensureExplicitCapacity(int minCapacity) { modCount++; // 將上面方法返回的大小和當前 elementData 大小進行比較,判斷是否需要擴容 if (minCapacity - elementData.length > 0) grow(minCapacity); } private void grow(int minCapacity) { // 具體擴容代碼 int oldCapacity = elementData.length; // 擴容后容量總是擴容前的大約1.5倍左右增量0.5 int newCapacity = oldCapacity + (oldCapacity >> 1); if (newCapacity - minCapacity < 0) newCapacity = minCapacity; // 如果擴容后超出了最大個數限制 則由hugeCapacity()來處理 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); } private static int hugeCapacity(int minCapacity) { if (minCapacity < 0) // overflow throw new OutOfMemoryError(); return (minCapacity > MAX_ARRAY_SIZE) ? Integer.MAX_VALUE : MAX_ARRAY_SIZE; }2.手動擴容或者指定初始容量
ArrayList 手動擴容主要是調用ensureCapacity(int minCapacity)方法,除此之外還能在創建ArrayList時指定初始容量
創建時直接指定容量new ArrayList<>(initialCapacity)及其實現
public ArrayList(int initialCapacity) { if (initialCapacity > 0) { // 創建指定容量的一個數組 this.elementData = new Object[initialCapacity]; } else if (initialCapacity == 0) { this.elementData = EMPTY_ELEMENTDATA; } else { throw new IllegalArgumentException("Illegal Capacity: "+ initialCapacity); } }通過ensureCapacity(int minCapacity)方法在處理過程中擴容,這種情況一般發生在處理過程中發現初始容量不能滿足當前數據需求,而又為了避免自動擴容時的資源浪費,因為每次自動擴容時都會進行數組復制。
public void ensureCapacity(int minCapacity) { int minExpand = (elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA) ? 0: DEFAULT_CAPACITY; if (minCapacity > minExpand) { ensureExplicitCapacity(minCapacity); } }3.最后看看擴容時的數組復制,數組復制是通過Arrays.copyOf(origin,newLenght)方法來實現的
public staticT[] copyOf(T[] original, int newLength) { return (T[]) copyOf(original, newLength, original.getClass()); } // 具體復制代碼 public static T[] copyOf(U[] original, int newLength, Class extends T[]> newType) { @SuppressWarnings("unchecked") T[] copy = ((Object)newType == (Object)Object[].class) ? (T[]) new Object[newLength] : (T[]) Array.newInstance(newType.getComponentType(), newLength); // 將舊數組中的數據復制到新數組 System.arraycopy(original, 0, copy, 0, Math.min(original.length, newLength)); return copy; }
ArrayList在方法中傳遞時,傳遞的是對象的引用,當某一個方法修改了ArrayList時,該修改會反映到引用該ArrayList的所有地方,不管傳遞多少次,都引用的時同一個內存區域的數據,所以如果要實現在傳遞時確保內容不變,應該克隆(用ArrayList的clone())方法一份進行傳遞(需要注意深淺克隆問題深克隆可以使用Collections.copy(dest,src)方法),具體該怎么弄,還是要視需求而定。7.ArrayList 的capacity和size的關系
如果要找他們之間的關系可能就是size=元素數量 <= capacity = 容量大小吧。size 反映的是當前數組中存了多少數據,而capacity則反映的是ArrayList內部數組的容量。不能習慣性認為capacity 既是 size8.ArrayList 和 LinkedList的選擇性問題
ArrayList和LinkedList為什么會存在選擇性問題,因為他們在get、add、remove時性能是不一樣的。9.為什么elementData使用transient修飾ArrayList是內部實現是數組,在隨機存取時時間復雜度是O(1)知道索引就能立馬取值,而其插入和刪除操作就相對比較麻煩,需要進行移位操作線性的時間復雜度。
LinkedList 是雙向連表Doubly-linked ,在插入和刪除時時間復雜度都是O(1),但索引(取索引位上的值)時需要從表頭或者表尾進行遍歷操作。
所以選用哪一個,完全取決于你要進行的操作是以隨機存取為主還是增刪元素較多為主。
transient關鍵字的作用是阻止對象序列化,那么為什么要防止elementData序列化呢?那是因為elementData是一個數組,并且并不是數組中每一個位置上都保存有值,容量10000的數組中可能只保存了10個對象。所以不能對其進行序列化,在ArrayList中重寫了writeObject 、readObject 方法來對ArrayList進行序列化控制。10.ArrayList實現RandomAccess接口有什么用
RandomAccess接口是一個標記接口并沒有定義任何方法,ArrayList 實現它是標記ArrayList支持快速隨機訪問這一特性!11.并發修改異常ConcurrentModificationException
并發修改異常ConcurrentModificationException通常出現在對一個List進行遍歷時,對正在遍歷的數組進行了修改性操作(修改性操作:改變大小(size=數量)的操作,而不是指具體值)(add,remove,clear等)時便會拋出這個異常,在ArrayList內部有一個protected transient int modCount = 0的變量用于記錄對ArrayList的修改,比如add方法代碼
public boolean add(E e) { ensureCapacityInternal(size + 1); // Increments modCount!! elementData[size++] = e; return true; } private void ensureCapacityInternal(int minCapacity) { ensureExplicitCapacity(calculateCapacity(elementData, minCapacity)); } private void ensureExplicitCapacity(int minCapacity) { modCount++; if (minCapacity - elementData.length > 0) grow(minCapacity); }可以看到在調用add方法時會執行modCount++操作以此標記最新修改的數量modCount
而在遍歷時比如forEach先看代碼
public void forEach(Consumer super E> action) { Objects.requireNonNull(action); final int expectedModCount = modCount; @SuppressWarnings("unchecked") final E[] elementData = (E[]) this.elementData; final int size = this.size; for (int i=0; modCount == expectedModCount && i < size; i++) { action.accept(elementData[i]); } if (modCount != expectedModCount) { throw new ConcurrentModificationException(); } }可以看到for循環中一個明確的條件是modCount == expectedModCount 每次遍歷都會檢測該條件是否成立,而在進入該段代碼之前先用final int expectedModCount = modCount;來保存代碼執行之前的修改數量,當進入遍歷后,有了更改操作,就會使得expectedModCount 和modCount不相等此時便會拋出ConcurrentModificationException異常,對于其他的修改操作,原理都是類似的。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/74311.html
摘要:當多個線程對同一個集合的內容進行操作時,就可能會產生事件。當某一個線程遍歷的過程中,的內容被另外一個線程所改變了就會拋出異常,產生事件。在線程在遍歷過程中的某一時刻,線程執行了,并且線程刪除了中的節點。 概要 前面,我們已經學習了ArrayList。接下來,我們以ArrayList為例,對Iterator的fail-fast機制進行了解。 1 fail-fast簡介 fail-fast...
摘要:集合集合類存放于包中。迭代器,可以通過迭代器遍歷集合中的數據是映射表的基礎接口有序集合的是非常常用的數據類型。按照指定的迭代器所返回的元素順序,將該中的所有元素添加到此列表的尾部。 集合集合類存放于Java.util包中。集合類型主要有3種:set(集)、list(列表包含Queue)和map(映射)。 Collection:Collection是集合的基本接口,List、Set、Qu...
摘要:本文是作者自己對中線程的狀態線程間協作相關使用的理解與總結,不對之處,望指出,共勉。當中的的數目而不是已占用的位置數大于集合番一文通版集合番一文通版垃圾回收機制講得很透徹,深入淺出。 一小時搞明白自定義注解 Annotation(注解)就是 Java 提供了一種元程序中的元素關聯任何信息和著任何元數據(metadata)的途徑和方法。Annotion(注解) 是一個接口,程序可以通過...
摘要:把內存分成兩種,一種叫做棧內存,一種叫做堆內存在函數中定義的一些基本類型的變量和對象的引用變量都是在函數的棧內存中分配。堆內存用于存放由創建的對象和數組。 一次慘痛的阿里技術面 就在昨天,有幸接到了阿里的面試通知,本來我以為自己的簡歷應該不會的到面試的機會了,然而機會卻這么來了,我卻沒有做好準備,被面試官大大一通血虐。因此,我想寫點東西紀念一下這次的經歷,也當一次教訓了。其實面試官大大...
閱讀 1268·2023-04-26 01:38
閱讀 1469·2021-11-15 11:39
閱讀 3259·2021-09-22 15:43
閱讀 2653·2019-08-30 15:55
閱讀 2057·2019-08-30 14:17
閱讀 2859·2019-08-29 14:16
閱讀 3071·2019-08-26 18:36
閱讀 2616·2019-08-26 12:19