學習一下Java的ArrayList和contains函數和擴容機制

来源:https://www.cnblogs.com/joey-redfield/archive/2023/10/25/17787980.html
-Advertisement-
Play Games

起因 在Leetcode上做題寫了兩種暴力解法,但是執行效率上不太一樣。 時間上差很遠,記憶體雖然差不多但是前者擊敗30%,後者擊敗94%。這兩種解法區別是用一條ArrayList還是兩條來存數據,所以contains雖然執行次數一樣但是檢測的長度上不一樣,而且ArrayList的擴容次數也不一樣,所 ...


起因

在Leetcode上做題寫了兩種暴力解法,但是執行效率上不太一樣。
image

時間上差很遠,記憶體雖然差不多但是前者擊敗30%,後者擊敗94%。這兩種解法區別是用一條ArrayList還是兩條來存數據,所以contains雖然執行次數一樣但是檢測的長度上不一樣,而且ArrayList的擴容次數也不一樣,所以學習一下。

contains(Object o)

直接翻(JDK8)源碼:
image
nullobject區分開來還是因為equals有一方是null的話都會導致異常. 合併一起寫的話可以用Objects.equals(obj1, obj2)的寫法.
所以顯然暴力解法用到的contains原理就是朴實無華的一遍遍搜索所以時間特別長.

ArrayList擴容機制

省流: 直接看最下麵的grow函數.

如果是預設的ArrayList, 添加元素時會先計算數組長度, 如果元素個數+1大於當前數組長度+1大於elementData.length時進行擴容,擴容後的數組大小是: oldCapacity + (oldCapacity >> 1) 可以理解成1.5倍擴容。

涉及到的源碼:

// 向指定索引位置插入元素
public void add(int index, E element) {
    // 檢查索引範圍
    rangeCheckForAdd(index);

    // 確保容量足夠
    ensureCapacityInternal(size + 1);  // 增加 modCount(用於支持併發修改的計數器)
    // 使用 System.arraycopy 將元素後移,為新元素騰出位置, 這是跟另一個add的區別⭐⭐⭐⭐⭐
    System.arraycopy(elementData, index, elementData, index + 1, size - index);
    elementData[index] = element; // 在指定位置插入新元素
    size++; // 更新列表大小
}

// 在列表末尾添加元素
public boolean add(E e) {
    // 確保容量足夠
    ensureCapacityInternal(size + 1);  // 增加 modCount
    elementData[size++] = e; // 在列表末尾添加新元素
    return true;
}

// 內部方法:確保容量足夠
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);
    }
    return minCapacity; // 否則返回所需容量
}

// 內部方法:確保容量足夠
private void ensureExplicitCapacity(int minCapacity) {
    modCount++; // 增加併發修改計數器

    // 檢查容量是否足夠,如果不夠則擴展
    if (minCapacity - elementData.length > 0)
        grow(minCapacity);
}

// 內部方法:擴展容量
private void grow(int minCapacity) {
    // 溢出安全的代碼
    int oldCapacity = elementData.length;
    int newCapacity = oldCapacity + (oldCapacity >> 1); // 新容量通常為舊容量的1.5倍
    if (newCapacity - minCapacity < 0)
        newCapacity = minCapacity; // 如果新容量小於所需容量,使用所需容量
    if (newCapacity - MAX_ARRAY_SIZE > 0)
        newCapacity = hugeCapacity(minCapacity); // 處理可能的巨大容量情況
    // 使用 Arrays.copyOf 擴展數組容量
    elementData = Arrays.copyOf(elementData, newCapacity);
}


實際上Array.copyof底層調用的還是System.arraycopy.


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

-Advertisement-
Play Games
更多相關文章
  • VitePress部署到Github Pages後發現樣式全錯亂怎麼辦? 當我們部署到Github pages線上後,發現全是樣式錯亂的,也就是無樣式,怎麼辦?在此簡單記錄一下 這個時候我們作為前端開發者,可以打開瀏覽器調試看看就發現了,是靜態資源地址不對,如下 這個時候,我們只需修改theme/c ...
  • 本文是Util應用框架 Angular UI 開發快速入門教程. Util前端技術概述 Util 應用框架目前僅支持用於開發管理後臺的 UI. 本文介紹了 Util UI 的技術特點和功能支持. UI 技術選型 Js語言 TypeScript TypeScript 是 微軟開發的腳本語言, 擴展了弱 ...
  • 說明 實測下載後的文件與源文件哈希值一致,保證數據傳輸安全一致。 如果下載到的文件每次都165KB左右,和源文件大小不符合,需要用IDE打開下載的文件,看看是否報致命錯誤,提示超過最大記憶體限制。這個與php.ini中的“memory_limit”參數配置有關,所以方法的$kilobyte參數不要設置 ...
  • 隨著社交媒體的快速發展,微博已成為了人們獲取信息的重要途徑。而在微博中,用戶和話題的排行榜更是引起了人們的廣泛關註。那麼如何獲取微博用戶和話題排行榜呢?下麵介紹一下基於微博排行榜API介面的方法。 一、獲取微博用戶排行榜API介面 微博用戶排行榜API介面是一種用於獲取微博用戶排名的介面。我們可以使 ...
  • Synchronized Synchronized關鍵字回顧 synchronized是java中的關鍵字,是一種同步鎖。它修飾的對象有以下幾種: 1.修飾一個代碼塊,被修飾的代碼塊稱為同步代碼塊,其作用的範圍是大括弧{},括起來的代碼,作用的對象是調用這個代碼塊的對象,synchronized不能 ...
  • 1. BaseHTTPRequestHandler介紹 BaseHTTPRequestHandler是Python中的一個基類,屬於http.server模塊,用於處理HTTP請求的基本功能。它提供了處理常見HTTP請求方法(如GET、POST等)的預設實現,並允許你在子類中進行定製化擴展。下麵詳細 ...
  • Dart語言和其他面向對象語言一樣,子類可以繼承父類,獲得父類的屬性和方法。那麼Dart語言,類繼承還有什麼特性呢…… ...
  • 源碼位置: /Lib/zipfile.py/ZipFile/_extract_member/zipfile.py或者直接點擊extract函數. 在使用python解壓縮zip文件時, 由於需要在解壓時重命名文件為我想要的格式, 而不巧的是, zipfile包官方源代碼沒有這個功能... 於是, 在 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...