ArrayList類源碼解析——ArrayList動態數組的實現細節(基於JDK8)

来源:https://www.cnblogs.com/jianguan/archive/2019/04/17/10707419.html
-Advertisement-
Play Games

通過源碼,分析了ArrayList類的繼承實現結構,主要對ArrayList動態數組數據結構的具體實現細節進行分析 ...


一、基本概念

 

    ArrayList是一個可以添加對象元素,併進行元素的修改查詢刪除等操作的容器類。ArrayList底層是由數組實現的,所以和數組一樣可以根據索引對容器對象所包含的元素進行快速隨機的查詢操作,其時間複雜度為O(1)。但是和數組不同的是,數組對象創建後數組長度是不變的,ArrayList對象創建後其長度是可變的,所以ArrayList也稱為動態數組,那麼ArrayList的動態數組數據結構又是如何實現的呢?接下來打開ArrayList源碼看看。

 

二、源碼分析

 

2.1、ArrayList類的繼承與實現

 

圖2.1 ArrayList類的繼承與實現(不包括繼承ArrayList的類)

  • 從繼承介面可以發現,ArrayList繼承了AbstractCollection抽象類,實現了List介面,並且實現還實現了RandomAccess,Cloneable和Serializable三個標記介面,所以ArrayList的的對象具有能根據索引對元素進行快速隨機訪問、可以調用Object的clone()方法、可以進行序列化的特性。有關Java的標記介面是什麼,在上一篇文章

         《Java的四個標記介面Serializable、Cloneable、RandomAccess和Remote介面》 進行了總結

 

2.2、ArrayList的屬性

 

  • 通過源碼可以知道ArrayList主要有以下屬性
 private static final long serialVersionUID = 8683452581122892189L;//用於ArrayList對象序列化的版本ID

 private static final int DEFAULT_CAPACITY = 10;//ArrayList對象的預設容量大小

 private static final Object[] EMPTY_ELEMENTDATA = {};//用於Null的ArrayList對象
   
 private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};//用於創建一個預設容量大小的ArrayList對象

 transient Object[] elementData; //用於創建動態大小的ArrayList對象,這個實例變數引用的數組可以說就是ArrayList容器
 
 private int size;//ArrayList對象的尺寸

 

 

2.3、ArrayList類的三個構造函數

 

一個無參構造函數,一個傳入一個int 量指定容量的構造函數和一個傳入一個容器對象來進行初始化的構造函數

 

    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);
        }
    }
//指定ArrayList初始容量的構造器

    public ArrayList() {
        this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
    }
//一個預設容量的ArrayList無參構造器
   
    public ArrayList(Collection<? extends E> c) {
        elementData = c.toArray();
        if ((size = elementData.length) != 0) {
          
            if (elementData.getClass() != Object[].class)
                elementData = Arrays.copyOf(elementData, size, Object[].class);
        } else {
            this.elementData = EMPTY_ELEMENTDATA;
        }
    }
//一個參數為一個容器對象的構造器,將該容器對象轉化為一個數組對象後,將ArrayList對象內的數組引用elementData初始化為這個數組對象(Object[])

 

2.4、ArrayList動態數組的具體實現細節

 

ArrayList對象創建後,是怎麼改變這個對象的容量實現動態數組的呢,來看看動態數組實現的幾個重要方法

 

  • grow方法實現了ArrayList對象容量增加的方式,一般ArrayList新的容量為原容量的1.5倍大小,源碼如下
  private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;//int整型的最大值減8
private void grow(int minCapacity) {
    
        int oldCapacity = elementData.length;//原ArrayList對象的容量
        int newCapacity = oldCapacity + (oldCapacity >> 1);//新的容量變為原容量+原容量/2,也就是新的容量增大為原來的1.5倍
        if (newCapacity - minCapacity < 0)
            newCapacity = minCapacity;
//新的容量如果還是小於指定容量,則新容量的值更新為指定容量
        if (newCapacity - MAX_ARRAY_SIZE > 0)
            newCapacity = hugeCapacity(minCapacity);
//如果新容量大於int整數的最大值減8的值,則調用hugeCapacity(minCapacity),新的容量為Interger.MAX_VALUE,即最大的Int整數
        elementData = Arrays.copyOf(elementData, newCapacity);
//創建一個數組長度為newCapacity,包含所有原ArrayList內elementData所有元素的新的elementDate數組
  

  •  hugeCapacity方法,在grow方法中被調用
private static int hugeCapacity(int minCapacity) {
        if (minCapacity < 0) // overflow
            throw new OutOfMemoryError();
        return (minCapacity > MAX_ARRAY_SIZE) ?
            Integer.MAX_VALUE :
            MAX_ARRAY_SIZE;
    }
View Code

 

  •  trimToSize方法。將ArrayList對象內的elementData對象的長度變為size,size變數保存了ArrayList對象所含元素個數。使用這個方法可以使ArrayList對象內的elementData數組長度為元素個數,以避免不必要的記憶體占用。
  • public void trimToSize() {
            modCount++;
            if (size < elementData.length) {
                elementData = (size == 0)? EMPTY_ELEMENTDATA: Arrays.copyOf(elementData, size);
            }
    //size小於elementData,如果size=0,令elementData為一個空數組;size>0,使elementDate數組變數指向包含原elememtData所有元素,且長度為元素個數size的新數組 }
     

 

  • ensureExplicitCapacity方法,在參數minCapacity大於elementDate引用的數組的長度時,調用grow方法來增加容量
private void ensureExplicitCapacity(int minCapacity) {
        modCount++;
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
    }

 

  • ensureCapacity方法,如果構造ArrayList對象時調用的是無參構造器,此時elementDate數組還沒創建,這個方法調用後可以確保ArrayList對象容量至少為10(預設值),且不小於minCapacity參數指定的值
    public void ensureCapacity(int minCapacity) {
        int minExpand = (elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA) ? 0: DEFAULT_CAPACITY;
        if (minCapacity > minExpand) {
            ensureExplicitCapacity(minCapacity);
        }
    }

 

  • ensureCapacityInternal方法,在調用add方法增加元素時,調用這個方法來確保元素個數+1(size+1)小於或等於elementDate數組的長度,若size+1大於element.length,在ensureExplicitCapacity方法中會調用grow方法對ArrayList對象進行擴容
 private void ensureCapacityInternal(int minCapacity) {
        ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));

}

  private static int calculateCapacity(Object[] elementData, int minCapacity) {
           if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
           return Math.max(DEFAULT_CAPACITY, minCapacity);

/*Arraylist類調用無參構造器初始化後,elementData數組初始化為DEFAULTCAPACITY_EMPTY_ELEMENTDATA(一個空對象數組Object{}),若elementDate為初始化的值這裡返回的是預設容量和minCapacity中的較大值*/
     }
           return minCapacity;//若elementDate不是初始化的值,這個方法直接返回minCapacity的值
  }

 

  •  add(int index)方法,實現了在指定的索引處添加元素,時間複雜度為O(n)
public void add(int index, E element) {
        rangeCheckForAdd(index);//先檢測索引是否越界

        ensureCapacityInternal(size + 1); //確保ArrayList的容量,即elementData數組的長度,在添加元素前大於size+1
        System.arraycopy(elementData, index, elementData, index + 1,
                         size - index);
//將從index處以及後面的元素的都向後移動一位,索引+1。所以調用這個方法的時間複雜度為O(n)
        elementData[index] = element;
//將元素element添加到索引為index的位置
        size++;//元素個數加一
    }

 

  •  remove(int index)方法,實現了在指定的位置刪除元素,時間複雜度為O(n)
 public E remove(int index) {
        rangeCheck(index);//檢查索引是否越界

        modCount++;
        E oldValue = elementData(index);

        int numMoved = size - index - 1;//這個元素後面的元素個數
        if (numMoved > 0)
            System.arraycopy(elementData, index+1, elementData, index,  numMoved);
//將這個元素後面的元素的前移一位,索引減一,原來index處的元素(要被刪除的元素)被後面一位的元素覆蓋。所以這個方法的時間複雜度為O(n) elementData[--size] = null;

//數組索引為size處沒有元素,使其為空,元素個數size-1 return oldValue;
//將刪除的元素作為方法的返回值 }

(ArrayList還有很多其他的方法,以上是ArrayList實現動態數組的主要方法,ArrayList類其他的方法在這裡就不一一贅述了)

 

(小官原創,若有謬誤,望各位大佬批評指正)

(轉載請註明出處)


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 今天學習了結構體這一章節,瞭解到了結構體在分配記憶體的時候採取的是對齊的方式 例如: 在這段程式中 輸出的並不是 15 //結構體集合內元素大小的簡單相加 而是 16 //此處就體現了在c語言結構體中記憶體開闢的原則 對齊原則 結構體對齊: 以最大的基本元素的大小(除數組)對齊,以本段程式為例: 1. ...
  • JANA面向對象的三大特性:封裝,繼承,多態。 今天學了繼承,繼承,通俗點說就是子類可以用父類的代碼,或重寫父類的方法、構造方法、屬性 例如我這裡要調用父類的方法: 下邊有兩個測試類,自己分別試一下,自己體驗效果。嘻嘻!!! 這是用父類new一個子類 這是直接new一個子類,這個子類的方法名如果和父 ...
  • CopyOnWriteArraySet是用Map實現的嗎? CopyOnWriteArraySet是有序的嗎? CopyOnWriteArraySet以何種方式保證元素不重覆? 如何比較兩個Set中的元素是否完全一致? ...
  • 一、各Set實現類的性能分析 HashSet和TreeSet是Set的兩個典型實現。HashSet的性能總是比TreeSet好(特別是最常用的添加、查詢元素等操作),因為TreeSet需要額外的紅黑樹演算法來維護集合元素的次序。只有當需要一個排序的Set時,才應該使用TreeSet,否則都應該使用Ha ...
  • 前面介紹了利用文件寫入器和文件讀取器來讀寫文件,因為FileWriter與FileReader讀寫的數據以字元為單位,所以這種讀寫文件的方式被稱作“字元流I/O”,其中字母I代表輸入Input,字母O代表輸出Output。可是FileWriter的讀操作並不高效,緣由在於FileWriter每次調用 ...
  • 索引 一、富文本編輯器 1.1 在Admin中使用 1.2 自定義使用 1.3 顯示 二、全文檢索 2.1 創建引擎及索引 2.2 使用 三、發送郵件 一、富文本編輯器 藉助富文本編輯器,網站的編輯人員能夠像使用offfice一樣編寫出漂亮的、所見即所得的頁面。此處以tinymce為例,其它富文本編 ...
  • 第二周-第02章節-Python3.5-模塊初識 G:\Python3.7.3\python.exe G:/practise/oldboy/day2/sys.py['G:/practise/oldboy/day2/sys.py'] 驅動器 G 中的捲沒有標簽。 捲的序列號是 C038-3181 G: ...
  • 文章大綱 一、Spring介紹二、Spring的IoC實戰三、IoC常見註解總結四、項目源碼及參考資料下載五、參考文章 一、Spring介紹 1. 什麼是Spring Spring是分層的Java SE/EE應用 full-stack輕量級開源框架,以IoC(Inverse Of Control:反 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...