日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

ArrayList的底层实现原理

發布時間:2025/3/15 编程问答 20 豆豆
生活随笔 收集整理的這篇文章主要介紹了 ArrayList的底层实现原理 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

ArrayList源碼分析

1、java.util.ArrayList<E> : List 接口的大小可變數組的實現類

  • ArrayList 內部基于 數組 存儲 各個元素。
  • 所謂大小可變數組,是指當 數組容量不足以存放新的元素時,創建新數組,并將原數組中的內容復制過來。

2、ArrayList底層實現原理

  • 構造方法源碼分析

    //對象數組:ArrayList的底層數據結構,transient表示該字段不進行序列化操作 transient Object[] elementData; //實例化一個空數組 private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; //實例化一個空數組 private static final Object[] EMPTY_ELEMENTDATA = {};/***指定初始容量的有參構造*/ public ArrayList(int initialCapacity) {//如果初始容量大于0就,對象數組就指向到一個新的數組,大小為所指定的大小if (initialCapacity > 0) {this.elementData = new Object[initialCapacity];} else if (initialCapacity == 0) {//如果大于0就指向一個空數組this.elementData = EMPTY_ELEMENTDATA;} else {throw new IllegalArgumentException("Illegal Capacity: "+initialCapacity);} } /** * 無參構造 */ public ArrayList() {//對象數組就指向到一個空數組this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA; }

    ArrayList基于數組實現,構造方法有有參構造和無參構造如果指定了初始容量且大于0就將對象數組指定到一個新的數組,大小為所指定的大小。如果調用無參構造就將對象數組指定到一個空的數組。

  • 添加方法源碼分析

    //elementData中已存放的元素的個數,注意:不是elementData的容量 private int size; //elementData的默認容量為10 private static final int DEFAULT_CAPACITY = 10; //對象數組:ArrayList的底層數據結構,transient表示該字段不進行序列化操作 transient Object[] elementData; //實例化一個空數組 private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; //實例化一個空數組 private static final Object[] EMPTY_ELEMENTDATA = {};protected transient int modCount = 0;@Native public static final int MAX_VALUE = 0x7fffffff;private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;/*** 向elementData末尾中添加元素*/ public boolean add(E e) {//確保對象數組elementData有足夠的容量,可以將新加入的元素e加進去ensureCapacityInternal(size + 1); //加入新元素e,size加1elementData[size++] = e;return true; }// minCapacity = seize+1,即表示執行完添加操作后,數組中的元素個數 private void ensureCapacityInternal(int minCapacity) {//判斷是否是空數組if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {//用最小容量和10進行比較,取最大值賦值給最小容量minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);}ensureExplicitCapacity(minCapacity);}/***確保數組的容量足夠存放新加入的元素,若不夠,要擴容*/ private void ensureExplicitCapacity(int minCapacity) {modCount++;//如果數組個數minCapacity (size+1)大于數組長度就需要進行擴容if (minCapacity - elementData.length > 0)grow(minCapacity); }private void grow(int minCapacity) {int oldCapacity = elementData.length;// 將舊的數組容量增加為原來的1.5倍作為新的容量int newCapacity = oldCapacity + (oldCapacity >> 1); //如果新的容量小于數組個數,將數組個數賦值給新容量if (newCapacity - minCapacity < 0)newCapacity = minCapacity;// 如果新的容量大于最大容量,就根據數組個數來決定新的容量大小if (newCapacity - MAX_ARRAY_SIZE > 0)newCapacity = hugeCapacity(minCapacity);// 根據新的容量,將數組拷貝到新的數組并賦值給數組elementData = Arrays.copyOf(elementData, newCapacity); }private static int hugeCapacity(int minCapacity) {// 如果數組個數小于0拋出OutOfMemoryError異常if (minCapacity < 0) // overflowthrow new OutOfMemoryError(); // 如果最數組個數大于最大容量 就返回最大值,否則返回最大容量return (minCapacity > MAX_ARRAY_SIZE) ?Integer.MAX_VALUE :MAX_ARRAY_SIZE; } /*** 向elementData指定位置添加元素*/ public void add(int index, E element) {//指定位置檢查rangeCheckForAdd(index);// 擴容檢查ensureCapacityInternal(size + 1); // Increments modCount!!//通過拷貝使數組內位置為 index 到 (size-1)的元素往后移動一位System.arraycopy(elementData, index, elementData, index + 1,size - index); // 找到位置添加元素elementData[index] = element;// 元素個數加一size++; } // 判斷指定位置是否超出數組個數 private void rangeCheckForAdd(int index) {if (index > size || index < 0)throw new IndexOutOfBoundsException(outOfBoundsMsg(index)); } /*** @param src 原數組.* @param srcPos 原數組的起始位置.* @param dest 目標數組.* @param destPos 目標數組的起始位置.* @param length 拷貝長度.*/ public static native void arraycopy(Object src, int srcPos,Object dest, int destPos,int length);

    ArrayList在無參的add方法中,每次插入新的元素時,先判斷是否需要擴容,判斷數組是否為空,為空則建默認的容量10賦值給minCapacity,判斷是否要擴容,第一次添加,數組的size是10,之后添加元素時,在ensureExplicitCapacity方法中判斷數組元素個數即size+1(形參minCapacity)是否超過數組長度,超過則需要進行擴容操作,擴容是將舊的容量擴大到1.5倍,然后將數組拷貝到新的數組完成擴容操作。最后將元素添加,并size+1。ArrayList在指定位置添加元素時,是先檢查指定位置是否在數組范圍內,即數組中元素個數是1,則index得小于或者低于1。然后通過拷貝使數組內位置為 index 到 (size-1)的元素往后移動一位,騰出位置之后放入元素,數組個數加一。

  • 刪除方法源碼分析

    public E remove(int index) {//指定位置檢查rangeCheck(index);modCount++; //保留要刪除的值E oldValue = elementData(index);//要移動元素個數int numMoved = size - index - 1;if (numMoved > 0)//通過拷貝使數組內位置為 index+1到 (size-1)的元素往前移動一位System.arraycopy(elementData, index+1, elementData, index,numMoved);//清除末尾元素讓GC回收elementData[--size] = null; // clear to let GC do its work//返回刪除的值return oldValue;}

    刪除指定位置的元素,先對index進行檢查,在將要刪除的值保留,計算出需要移動元素個數,再通過拷貝使數組內位置為 index+1到 (size-1)的元素往前移動一位,最后將末尾元素清理讓GC回收返回刪除值。由于List接口繼承了Collection,因此ArrayList還有一個來自Collection接口定義的刪除對象boolean remove( Object o ) ,而ArrayList自定義的remove方法是T remove(int index)刪除的是下標位置的對象并返回值。

總結

以上是生活随笔為你收集整理的ArrayList的底层实现原理的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。