迎難而上ArrayList,源碼分析走一波

来源:https://www.cnblogs.com/qing-gee/archive/2020/07/12/13286907.html
-Advertisement-
Play Games

先看再點贊,給自己一點思考的時間,思考過後請毫不猶豫微信搜索【沉默王二】,關註這個長髮飄飄卻靠才華苟且的程式員。本文 GitHub github.com/itwanger 已收錄,裡面還有技術大佬整理的面試題,以及二哥的系列文章。 關於 Java 基礎、Java 面向對象編程、Java 字元串、Ja ...


先看再點贊,給自己一點思考的時間,思考過後請毫不猶豫微信搜索【沉默王二】,關註這個長髮飄飄卻靠才華苟且的程式員。
本文 GitHub github.com/itwanger 已收錄,裡面還有技術大佬整理的面試題,以及二哥的系列文章。

關於 Java 基礎、Java 面向對象編程、Java 字元串、Java 數組等方面的知識點已經可以告一段落了,小伙伴們可以在「沉默王二」公眾號後臺回覆「小白」獲取第二版手冊。覺得不錯的話,請隨手轉發給身邊的小伙伴,贈人玫瑰,手有餘香哈。

那麼接下來,我開始肝 Java 集合方面的文章了,小伙伴們請默默為我鼓個掌,我能聽得到,真的,別吝嗇你的掌聲,響起來。第一篇,必須得從 ArrayList 開始,畢竟 ArrayList 可以稱得上是集合方面最常用的類了,估計沒有之一。

ArrayList 實現了 List 介面,是基於數組實現的。小伙伴們都知道,數組的大小是固定的,創建的時候指定了大小,就不能再調整了,如果數組滿了,就不能再添加任何元素了。ArrayList 是數組很好的替代方案,它提供了比數組更豐富的預定義方法(增刪改查),並且大小是可以根據元素的多少進行自動調整的,非常靈活。

準備在 ArrayList 的第四個位置(下標為 3)上添加一個元素 55。

此時 ArrayList 中第五個位置以後的元素將會向後移動。

準備把 23 從 ArrayList 中移除。

此時下標為 7、8、9 的元素往前挪。

01、如何創建一個 ArrayList

ArrayList<String> alist = new ArrayList<String>();

可以通過上面的語句來創建一個字元串類型的 ArrayList(通過尖括弧來限定 ArrayList 中元素的類型,如果嘗試添加其他類型的元素,將會產生編譯錯誤),更簡化的寫法如下:

List<String> alist = new ArrayList<>();

由於 ArrayList 實現了 List 介面,所以 alist 變數的類型可以是 List 類型;new 關鍵字聲明後的尖括弧中可以不再指定元素的類型,因為編譯器可以通過前面尖括弧中的類型進行智能推斷。

如果非常確定 ArrayList 中元素的個數,在創建的時候還可以指定初始大小。

List<String> alist = new ArrayList<>(20);

這樣做的好處是,可以有效地避免在添加新的元素時進行不必要的擴容。但通常情況下,我們很難確定 ArrayList 中元素的個數,因此一般不指定初始大小。

02、向 ArrayList 中添加一個元素

可以通過 add() 方法向 ArrayList 中添加一個元素,如果不指定下標的話,就預設添加在末尾。

alist.add("沉默王二");

感興趣的小伙伴可以研究一下 add() 方法的源碼,它在添加元素的時候會執行 grow() 方法進行擴容,這個是面試官特別喜歡考察的一個重點。

下麵是 add(E e) 方法的源碼:

public boolean add(E e) {
    modCount++;
    add(e, elementData, size);
    return true;
}

調用了私有的 add(E e, Object[] elementData, int s) 方法:

private void add(E e, Object[] elementData, int s) {
    if (s == elementData.length)
        elementData = grow();
    elementData[s] = e;
    size = s + 1;
}

然後調用了非常關鍵的 grow(int minCapacity) 方法:

private Object[] grow(int minCapacity) {
    int oldCapacity = elementData.length;
    if (oldCapacity > 0 || elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        int newCapacity = ArraysSupport.newLength(oldCapacity,
                minCapacity - oldCapacity, /* minimum growth */
                oldCapacity >> 1           /* preferred growth */);
        return elementData = Arrays.copyOf(elementData, newCapacity);
    } else {
        return elementData = new Object[Math.max(DEFAULT_CAPACITY, minCapacity)];
    }
}

如果創建 ArrayList 的時候沒有指定初始大小,那麼 ArrayList 的初始大小就是 DEFAULT_CAPACITY:

private static final int DEFAULT_CAPACITY = 10;

可以容納 10 個元素。

還可以通過 add(int index, E element) 方法把元素添加到指定的位置:

alist.add(0"沉默王三");

add(int index, E element) 方法的源碼如下:

public void add(int index, E element) {
    rangeCheckForAdd(index);
    modCount++;
    final int s;
    Object[] elementData;
    if ((s = size) == (elementData = this.elementData).length)
        elementData = grow();
    System.arraycopy(elementData, index,
            elementData, index + 1,
            s - index);
    elementData[index] = element;
    size = s + 1;
}

該方法會調用到一個非常重要的本地方法 System.arraycopy(),它會對數組進行複製(要插入位置上的元素往後複製,參照文章一開頭提到的兩張圖片)。

03、更新 ArrayList 中的元素

可以使用 set() 方法來更改 ArrayList 中的元素,需要提供下標和新元素。

alist.set(0"沉默王四");

原來 0 位置上的元素為“沉默王三”,現在將其更新為“沉默王四”。

來看一下 set() 方法的源碼:

public E set(int index, E element) {
    Objects.checkIndex(index, size);
    E oldValue = elementData(index);
    elementData[index] = element;
    return oldValue;
}

該方法會先對指定的下標進行檢查,看是否越界,然後替換新值並返回舊值。

04、刪除 ArrayList 中的元素

remove(int index) 方法用於刪除指定下標位置上的元素,remove(Object o) 方法用於刪除指定值的元素。

alist.remove(1);
alist.remove("沉默王四");

先來看 remove(int index) 方法的源碼:

public E remove(int index) {
    Objects.checkIndex(index, size);
    final Object[] es = elementData;

    @SuppressWarnings("unchecked") E oldValue = (E) es[index];
    fastRemove(es, index);

    return oldValue;
}

該方法返回要刪除的元素,真正的刪除操作在 fastRemove(es, index) 方法中。

再來看 remove(Object o) 方法的源碼:

public boolean remove(Object o) {
    final Object[] es = elementData;
    final int size = this.size;
    int i = 0;
    found: {
        if (o == null) {
            for (; i < size; i++)
                if (es[i] == null)
                    break found;
        } else {
            for (; i < size; i++)
                if (o.equals(es[i]))
                    break found;
        }
        return false;
    }
    fastRemove(es, i);
    return true;
}

該方法通過 break label 的方式找到要刪除元素(null 的時候使用 == 操作符判斷,非 null 的時候使用 equals() 方法,意味著如果有相同元素時,刪除第一個)的下標,然後調用 fastRemove() 方法。

既然都調用了 fastRemove() 方法,那就繼續來跟蹤一下源碼:

private void fastRemove(Object[] es, int i) {
    modCount++;
    final int newSize;
    if ((newSize = size - 1) > i)
        System.arraycopy(es, i + 1, es, i, newSize - i);
    es[size = newSize] = null;
}

當刪除的是末尾的元素時,不需要複製數組,直接把末尾的元素賦值為 null 即可;否則的話,就需要調用 System.arraycopy() 對數組進行複製。參照文章一開頭提到的第三張、第四張圖片。

05、查找 ArrayList 中的元素

如果要正序查找一個元素,可以使用 indexOf() 方法;如果要倒序查找一個元素,可以使用 lastIndexOf() 方法。

alist.indexOf("沉默王二");
alist.lastIndexOf("沉默王二");

來看一下 indexOf() 方法的源碼:

public int indexOf(Object o) {
    return indexOfRange(o, 0, size);
}

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

-Advertisement-
Play Games
更多相關文章
  • 最近一直被win10彈出的“修複你的工作或學校賬戶”煩擾, 點進去發現原來是之前公司的賬號還在win10下,和我的微軟賬戶綁定在一起, 百度了一堆都沒有可用的刪除方法。 最後發現在 <設置> > <連接工作或學校賬戶> 中有此公司賬號,並且下麵有一個按鈕<斷開連接>: 這裡斷開連接就行,我這裡已經刪 ...
  • 一、在canvas畫布中如何載入圖片 用drawImage( )方法 drawImage用法的三種情況: 1、在畫布指定位置定義圖像 ctx.drawImage(img,x,y); 註:此時畫布上顯示的圖片大小是圖片的預設大小 2、在畫布上定點陣圖像,並規定圖像的寬度和高度: ctx.drawImag ...
  • content方面 減少HTTP請求:合併文件、CSS精靈、inline Image 減少DNS查詢:DNS緩存、將資源分佈到恰當數量的主機名 減少DOM元素數量 Server方面 使用CDN 配置ETag 對組件使用Gzip壓縮 Cookie方面 減小cookie大小 css方面 將樣式表放到頁面 ...
  • sort()方法、 reverse()方法、數學對象MATH(常用)、計時器 ...
  • Web前端並不算新興行業,在國內也已經發展好些年了,隨著互聯網電商項目的不斷發展,讓企業越來越重視用戶體驗,而Web前端開發工程師正是實現這樣一切的關鍵。 Web前端工程師已成為目前行業內最緊俏的職位。Web前端開發依然是值得大家選擇的職業。各個企業對於人才稀缺量比較大,和其他的行業相比它還沒有達到 ...
  • 如果你是一位前端開發工程師,對“跨平臺”一詞應該不會感到陌生。像常見的前端框架:比如React、Vue、Angular,它們可以做網頁端,也可以做移動端,但很少能做到跨PC、Mac端,也就是我們熟知的Windows、Linux以及macOS上的應用程式。即使有,受限於JS的性能瓶頸,當有大量的科學計 ...
  • type 指示type要使用的驗證器。可識別的類型值為: string:類型必須為string。type 預設是 string // 校驗 string: [ {type: 'string', message: `請輸入字元串`, trigger: 'blur'} ] <el-form-item l ...
  • var arr = [4,0,7,9,0,0,2,6,0,3,1]; var x; function zy(arr,x) { for(var i=0; i<arr.length;i++) { if(arr[i] == x) { arr.splice(i,1); i--; } } } zy(arr,0 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...