欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java中的ArrayList容量及擴容方式

 更新時間:2021年09月13日 17:01:02   作者:zx2015216856  
這篇文章主要介紹了Java中的ArrayList容量及擴容方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教

查看JDK1.8 ArrayList的源代碼

1、默認初始容量為10

   /**
     * Default initial capacity.
     */
    private static final int DEFAULT_CAPACITY = 10;

2、最大容量為 Integer.MAX_VALUE - 8

    /**
     * The maximum size of array to allocate.
     * Some VMs reserve some header words in an array.
     * Attempts to allocate larger arrays may result in
     * OutOfMemoryError: Requested array size exceeds VM limit
     */
    private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

原因:之前參考別人的,有待求證:

數(shù)組對象有一個額外的元數(shù)據(jù),用于表示數(shù)組的大??;

數(shù)組長度size為int類型,共32位,有一位符號位,所以最大長度為Integer.MAX_VALUE=2^31= 2,147,483,648;

8bytes用來存儲size;

3、擴容方式:

(1)首先傳遞進來一個希望的最小容量minCapacity;

(2)新容量newCapacity = oldCapacity + (oldCapacity >> 1),即新容量等于原容量的1.5倍;

(3)如果minCapacity > newCapacity ,newCapacity = minCapacity ;

(4)如果 newCapacity > 最大容量 MAX_ARRAY_SIZE ,newCapacity = hugeCapacity(minCapacity);

(5)以新容量拷貝原數(shù)據(jù)

   /**
     * Increases the capacity to ensure that it can hold at least the
     * number of elements specified by the minimum capacity argument.
     *
     * @param minCapacity the desired minimum capacity
     */
    private void grow(int minCapacity) {
        // overflow-conscious code
        int oldCapacity = elementData.length;
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        if (newCapacity - minCapacity < 0)
            newCapacity = minCapacity;
        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;
    }

Java ArrayList() 擴容原理

平常都是直接使用 ArrayList(),今天特地看一下 ArrayList() 的擴容原理。

先看下 ArrayList 的屬性以及構(gòu)造方法,這個比較重要

先看下屬性

// ArrayList 默認容量大小
private static final int DEFAULT_CAPACITY = 10;
// 一個共享的空數(shù)組, 在空實例時使用
private static final Object[] EMPTY_ELEMENTDATA = {};
// 一個共享的空數(shù)組, 在使用默認 size 的空實例時使用
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
/*
存儲 ArrayList 元素的數(shù)組緩沖區(qū)
重點1: ArrayList 的容量是數(shù)組緩沖區(qū)的長度
重點2: 從這個元素也可以看的出來 ArrayList() 的底層就是一個 Object[]  
add 第一個元素時, 任何帶有 elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA 的空 ArrayList 都將被擴展為 DEFAULT_CAPACITY
*/
transient Object[] elementData;
// ArrayList 的大小, 我們平常使用的 list.size() 底層就是記錄的這個 size
private int size;
// ArrayList 最大
private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

再看構(gòu)造器,帶參構(gòu)造器

/*
帶參構(gòu)造器, 參數(shù)為容量大小, 例如: 初始化一個容器為 20 類型為 Integer 的 ArrayList
ArrayList<Integer> list = new ArrayList<>(20);
*/
public ArrayList(int initialCapacity) {
 /*
 初始化容量 > 0, elementData 初始化為初始化容量大小的數(shù)組
 初始化容量 = 0, elementData = EMPTY_ELEMENTDATA (空數(shù)組)
 初始化容量 < 0, 直接拋出異常
 */
    if (initialCapacity > 0) {
        this.elementData = new Object[initialCapacity];
    } else if (initialCapacity == 0) {
        this.elementData = EMPTY_ELEMENTDATA;
    } else {
        throw new IllegalArgumentException("Illegal Capacity: "+
                                           initialCapacity);
    }
}

參數(shù)為 Collection 的構(gòu)造器

/*
將一個參數(shù)為 Collection 的集合轉(zhuǎn)換為 ArrayList
*/
public ArrayList(Collection<? extends E> c) {
    // Collection 轉(zhuǎn)換為數(shù)組 Object[] 類型 
    elementData = c.toArray();
    // 判斷當前對象大小是否和 Collection 長度相等并且不等于 0, false 的話 elementData 等于空數(shù)組了
    if ((size = elementData.length) != 0) {
     // c.toArray() 可能不會正確地返回一個 Object[] 數(shù)組,所以使用 Arrays.copyOf()
        if (elementData.getClass() != Object[].class)
            elementData = Arrays.copyOf(elementData, size, Object[].class);
    } else {
        this.elementData = EMPTY_ELEMENTDATA;
    }
}

不帶參構(gòu)造器

/*
不帶參構(gòu)造器就像我們平時使用一樣, 直接 new 一個 ArrayList 不需要傳遞任何參數(shù)
構(gòu)造方法中直接將 elementData 初始化為 DEFAULTCAPACITY_EMPTY_ELEMENTDATA (空數(shù)組)
*/
public ArrayList() {
    this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
}

看到這里就發(fā)現(xiàn),帶參的構(gòu)造器我們可以直接傳遞參數(shù),而默認的構(gòu)造器怎么怎么樣初始化容量大小的呢?

add() 方法可以直接得到答案。

public boolean add(E e) {
 // 這一行是關鍵, 看下面
    ensureCapacityInternal(size + 1);
    // 將元素追加到集合的末尾 假如當前 size = 10 size++ 追加到第 11 位
    elementData[size++] = e;
    return true;
}

ensureCapacityInternal() 方法調(diào)用

private void ensureCapacityInternal(int minCapacity) {
 /*
 calculateCapacity() 方法, 剛初始化時會返回 10, 其他情況返回當前 size + 1
 ensureExplicitCapacity() 方法, 調(diào)用擴容
 */
    ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}
private static int calculateCapacity(Object[] elementData, int minCapacity) { 
 /*
 使用無參構(gòu)造器創(chuàng)建創(chuàng)建 ArrayList 的集合, 此時一定是相等的
 */
    if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
     /*
     兩數(shù)相比返回最大值, 此時 Math.max(10, 1); 
     默認容量, 由此而來
     */
        return Math.max(DEFAULT_CAPACITY, minCapacity);
    }
    // 不相等的話只有返回當前的 size + 1
    return minCapacity;
}
private void ensureExplicitCapacity(int minCapacity) {
    // 增量, 記錄修改/更新次數(shù)
    modCount++;  
    
     // 初始化: 10 - 0 > 0
     // 其他: size + 1 > 0
    if (minCapacity - elementData.length > 0)
     // 擴容操作
        grow(minCapacity);
}
private void grow(int minCapacity) {
    // 老的長度, 初始化時為 0, 
    int oldCapacity = elementData.length;
    // 新的長度此時 0 + (0 >> 1), newCapacity = 0
    int newCapacity = oldCapacity + (oldCapacity >> 1);
    // 初始化場景: 0 - 10 < 0 ? true newCapacity = 10
    if (newCapacity - minCapacity < 0)
        newCapacity = minCapacity;
    // 初始化場景: 10 - 2147483639 > 0 返回 false
    if (newCapacity - MAX_ARRAY_SIZE > 0)
     // 超大長度才可以執(zhí)行這個方法, 必須大于 MAX_ARRAY_SIZE 一般不會
        newCapacity = hugeCapacity(minCapacity);
    // 復制原數(shù)組中的元素, 并擴容
    elementData = Arrays.copyOf(elementData, newCapacity);
}

上看說的是初始化場景,下面看一下其他場景,也是相當簡單

private void ensureCapacityInternal(int minCapacity) {
 /*
 calculateCapacity() 方法, 正常擴容返回 size + 1, 比如 10 + 1, 因為默認長度為 10 當再次新增數(shù)據(jù)時就會出發(fā)擴容
 ensureExplicitCapacity() 方法, 調(diào)用擴容
 */
    ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}
private static int calculateCapacity(Object[] elementData, int minCapacity) { 
    if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        return Math.max(DEFAULT_CAPACITY, minCapacity);
    }
    /*
 elementData 不等于空數(shù)組
 返回當前的 size + 1, 即 10 + 1 返回 11
 */
    return minCapacity;
}
private void ensureExplicitCapacity(int minCapacity) {
    // 增量, 記錄修改/更新次數(shù)
    modCount++;  
    
    // 其他: 11 - 10 > 0 true, 觸發(fā)擴容, 如果當前下表是 5 的話 5 + 1 =6, 6 < 10 是, 此時不會出發(fā)擴容
    if (minCapacity - elementData.length > 0)
     // 擴容操作
        grow(minCapacity);
}
private void grow(int minCapacity) {
    // 老的長度 10
    int oldCapacity = elementData.length;
    // 新的長度此時 10 + (10 >> 1), newCapacity = 15
    int newCapacity = oldCapacity + (oldCapacity >> 1);
    // 15 - 11 < 0 ? false
    if (newCapacity - minCapacity < 0)
        newCapacity = minCapacity;
    // 15 - 2147483639 > 0 返回 false
    if (newCapacity - MAX_ARRAY_SIZE > 0)
     // 超大長度才可以執(zhí)行這個方法, 必須大于 MAX_ARRAY_SIZE 一般不會
        newCapacity = hugeCapacity(minCapacity);
    // 復制原數(shù)組中的元素, 并擴容 newCapacity = 15
    elementData = Arrays.copyOf(elementData, newCapacity);
}

結(jié)論

1、 觸發(fā)擴容的關鍵是

當前 size + 1 是否大于當前容量,如果大于容量則證明,集合不夠用了,需要擴容。如果小與當前容量則證明集合還有容量不需要擴容!

2、 每次擴容的大小是

oldCapacity + (oldCapacity >> 1) 即: 10 + (10 >> 1)

即:當前容量的 1.5 倍!

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關文章

  • idea中springboot整合mybatis找不到mapper接口的原因分析

    idea中springboot整合mybatis找不到mapper接口的原因分析

    這篇文章主要介紹了idea中springboot整合mybatis找不到mapper接口的原因分析及解決,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-01-01
  • 關于maven install報錯原因揭秘:parent.relativePath指向錯誤的本地POM文件

    關于maven install報錯原因揭秘:parent.relativePath指向錯誤的本地POM文件

    在使用Maven進行項目構(gòu)建時,如果遇到'parent.relativePath'指向錯誤的本地POM文件的問題,可能會導致構(gòu)建失敗,這通常是由于父項目POM文件的相對路徑設置錯誤、本地POM文件與父項目POM文件版本或內(nèi)容不一致所致,解決方法包括檢查并修正父項目POM文件中的相對路徑設置
    2024-09-09
  • 關于Java中配置ElasticSearch集群環(huán)境賬號密碼的問題

    關于Java中配置ElasticSearch集群環(huán)境賬號密碼的問題

    這篇文章主要介紹了Java中配置ElasticSearch集群環(huán)境賬號密碼的問題,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-04-04
  • Java代理模式與動態(tài)代理之間的關系以及概念

    Java代理模式與動態(tài)代理之間的關系以及概念

    代理模式是開發(fā)中常見的一種設計模式,使用代理模式可以很好的對程序進行橫向擴展。動態(tài)代理:代理類在程序運行時被創(chuàng)建的代理方式。關鍵在于動態(tài),程序具有了動態(tài)特性,可以在運行期間根據(jù)不同的目標對象生成動態(tài)代理對象
    2023-02-02
  • spring常用注解開發(fā)一個RESTful接口示例

    spring常用注解開發(fā)一個RESTful接口示例

    這篇文章主要為大家介紹了使用spring常用注解開發(fā)一個RESTful接口示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步
    2022-03-03
  • spring boot + mybatis如何實現(xiàn)數(shù)據(jù)庫的讀寫分離

    spring boot + mybatis如何實現(xiàn)數(shù)據(jù)庫的讀寫分離

    這篇文章主要給大家介紹了關于spring boot + mybatis如何實現(xiàn)數(shù)據(jù)庫的讀寫分離的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用spring boot具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧
    2019-09-09
  • CountDownLatch和Atomic原子操作類源碼解析

    CountDownLatch和Atomic原子操作類源碼解析

    這篇文章主要為大家介紹了CountDownLatch和Atomic原子操作類的源碼解析以及理解應用,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步
    2022-03-03
  • Spring中的FactoryBean實現(xiàn)原理詳解

    Spring中的FactoryBean實現(xiàn)原理詳解

    這篇文章主要介紹了Spring中的FactoryBean實現(xiàn)原理詳解,spring中有兩種類型的Bean,一種是普通的JavaBean,另一種就是工廠Bean(FactoryBean),這兩種Bean都受Spring的IoC容器管理,但它們之間卻有一些區(qū)別,需要的朋友可以參考下
    2024-02-02
  • Spring多種加載Bean方式解析

    Spring多種加載Bean方式解析

    本篇文章主要介紹了Spring多種加載Bean方式解析,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-04-04
  • java開發(fā)之SQL語句中DATE_FORMAT函數(shù)舉例詳解

    java開發(fā)之SQL語句中DATE_FORMAT函數(shù)舉例詳解

    要將日期值格式化為特定格式,請使用DATE_FORMAT函數(shù),下面這篇文章主要給大家介紹了關于java開發(fā)之SQL語句中DATE_FORMAT函數(shù)的相關資料,文中通過代碼介紹的非常詳細,需要的朋友可以參考下
    2024-05-05

最新評論