成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Java ArrayList源碼分析

DevTalking / 1058人閱讀

摘要:構(gòu)造器提供了三個(gè)構(gòu)造器接口約定,每個(gè)集合類應(yīng)該提供兩個(gè)標(biāo)準(zhǔn)構(gòu)造器,一個(gè)是無參數(shù)的構(gòu)造器上面第一個(gè),另外一個(gè)是擁有單個(gè)參數(shù)類型為的構(gòu)造器上面第二個(gè)。還提供了第三個(gè)構(gòu)造器,其接受一個(gè)值,用于設(shè)置的初始大小默認(rèn)大小為。

ArrayList介紹

List 接口的一個(gè)實(shí)現(xiàn)類,內(nèi)部是用一個(gè)數(shù)組存儲元素值,相當(dāng)于一個(gè)可變大小的數(shù)組。

簽名
public class ArrayList
extends AbstractList
implements List, RandomAccess, Cloneable, Serializable

可以看到ArrayList繼承了AbstractList抽象類,它實(shí)現(xiàn)了List接口的大多數(shù)方法。如果要實(shí)現(xiàn)一個(gè)不可變的List,只要繼承這個(gè)類并且實(shí)現(xiàn)get(int)size方法。如果要實(shí)現(xiàn)可變的List,需要覆蓋set(int, E)。另外,如果List的大小是可變的,還要覆蓋add(int, E)remove()方法。

構(gòu)造器

ArrayList提供了三個(gè)構(gòu)造器:

ArrayList()
ArrayList(Collection c)
ArrayList(int initialCapacity)

Collection接口約定,每個(gè)集合類應(yīng)該提供兩個(gè)”標(biāo)準(zhǔn)”構(gòu)造器,一個(gè)是無參數(shù)的構(gòu)造器(上面第一個(gè)),另外一個(gè)是擁有單個(gè)參數(shù)(類型為Collettion)的構(gòu)造器(上面第二個(gè))。ArrayList還提供了第三個(gè)構(gòu)造器,其接受一個(gè)int值,用于設(shè)置ArrayLi的初始大?。J(rèn)大小為10)。

相關(guān)方法 trimToSize
public void trimToSize() {
        modCount++;
        int oldCapacity = elementData.length;
        if (size < oldCapacity) {
            elementData = Arrays.copyOf(elementData, size);
        }
    }

用于把ArrayList的容量縮減到當(dāng)前實(shí)際大小,減少存儲容量。其中的變量modCountAbstracList繼承而來,記錄List發(fā)生結(jié)構(gòu)化修改(structurally modified)的次數(shù)。elementData中實(shí)際存儲了ArrayList的元素,在ArrayList中聲明為:private transient Object[] elementData;變量sizeArrayList的元素?cái)?shù)量,當(dāng)size < oldCapacity時(shí),調(diào)用Arrays.copyOf方法實(shí)現(xiàn)縮減。

indexOflasIndexOf
public int indexOf(Object o) {
        if (o == null) {
            for (int i = 0; i < size; i++)
                if (elementData[i]==null)
                    return i;
        } else {
            for (int i = 0; i < size; i++)
                if (o.equals(elementData[i]))
                    return i;
        }
        return -1;
    }

這兩個(gè)方法返回指定元素的下標(biāo),要區(qū)分參數(shù)是否為null。lastIndexOfindexOf類似,只不過是從后往前搜索。

ensureCapacity
public void ensureCapacity(int minCapacity) {
       if (minCapacity > 0)
           ensureCapacityInternal(minCapacity);
   }
private void ensureCapacityInternal(int minCapacity) {
        modCount++;
        // overflow-conscious code
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
    }
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);
    }

這個(gè)方法可以確保ArrayList的大小

addaddAll
public void add(int index, E element) {
        rangeCheckForAdd(index);

        ensureCapacityInternal(size + 1);  // Increments modCount!!
        System.arraycopy(elementData, index, elementData, index + 1,
                         size - index);
        elementData[index] = element;
        size++;
    }

add(int index, E element)向指定位置添加元素,首先調(diào)用rangeCheckForAdd檢查index是否有效,如果index > size || index < 0將拋出異常。然后確保容量加1,調(diào)用System.arraycopy把從index開始的元素往后移動一個(gè)位置。最后把index處的值設(shè)置為添加的元素。還有一個(gè)重載的add(E e)方法是直接把元素添加到末尾。
addAll(Collection c)addAll(int index, Collection c)則分別是向末尾和指定位置添加Collection中的所有元素。

removeremoveAll
public boolean remove(Object o) {
        if (o == null) {
            for (int index = 0; index < size; index++)
                if (elementData[index] == null) {
                    fastRemove(index);
                    return true;
                }
        } else {
            for (int index = 0; index < size; index++)
                if (o.equals(elementData[index])) {
                    fastRemove(index);
                    return true;
                }
        }
        return false;
    }

remove(Object o)方法刪除指定的元素。首先是查找元素位置,然后調(diào)用fastRemove(index)刪除,其代碼如下:

private void fastRemove(int index) {
        modCount++;
        int numMoved = size - index - 1;
        if (numMoved > 0)
            //把index+1往后的元素都往前移動一個(gè)位置
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        elementData[--size] = null; // Let gc do its work
    }

重載的remove(int index)方法用于刪除指定位置的元素。removeRange(int fromIndex, int toIndex)用于刪除指定位置之間的所有元素。
removeAll(Collection c)retainAll(Collection c)代碼如下:

public boolean removeAll(Collection c) {
        Objects.requireNonNull(c);
        return batchRemove(c, false);
    }
public boolean retainAll(Collection c) {
        Objects.requireNonNull(c);
        return batchRemove(c, true);
    }

它們都是通過調(diào)用batchRemove方法實(shí)現(xiàn)的,其代碼如下:

private boolean batchRemove(Collection c, boolean complement) {
        final Object[] elementData = this.elementData;
        int r = 0, w = 0;
        boolean modified = false;
        try {
            for (; r < size; r++)
                if (c.contains(elementData[r]) == complement)
                    elementData[w++] = elementData[r];
        } finally {
            // Preserve behavioral compatibility with AbstractCollection,
            // even if c.contains() throws.
            if (r != size) {
                System.arraycopy(elementData, r,
                                 elementData, w,
                                 size - r);
                w += size - r;
            }
            if (w != size) {
                // clear to let GC do its work
                for (int i = w; i < size; i++)
                    elementData[i] = null;
                modCount += size - w;
                size = w;
                modified = true;
            }
        }
        return modified;
    }

這個(gè)方法有兩個(gè)參數(shù),第一個(gè)是操作的Collection,第二個(gè)是一個(gè)布爾值,通過設(shè)置為truefalse來選擇是removeAll還是retainAll。try里面的語句是把留下來的放在0到w之間,然后在finally中第二個(gè)if處理w之后的空間,第一個(gè)是在c.contains()拋出異常時(shí)執(zhí)行。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/64527.html

相關(guān)文章

  • Java集合源碼分析系列-(一)ArrayList源碼剖析

    摘要:需要注意的是,通過構(gòu)造函數(shù)定義初始量是動態(tài)數(shù)組的實(shí)際大小。帶容量的構(gòu)造函數(shù)新建一個(gè)容量為的數(shù)組默認(rèn)構(gòu)造函數(shù),默認(rèn)為空構(gòu)造一個(gè)包含指定元素的第一個(gè)構(gòu)造方法使用提供的來初始化數(shù)組的大小。 前言 今天介紹經(jīng)常使用的一個(gè)Java集合類——ArrayList(基于JDK1.8.0_121)。ArrayList在工作和日常面試中經(jīng)常被使用或者提到。總的來說,工作中使用ArrayList主要是因?yàn)閯?..

    Miyang 評論0 收藏0
  • java源碼

    摘要:集合源碼解析回歸基礎(chǔ),集合源碼解析系列,持續(xù)更新和源碼分析與是兩個(gè)常用的操作字符串的類。這里我們從源碼看下不同狀態(tài)都是怎么處理的。 Java 集合深入理解:ArrayList 回歸基礎(chǔ),Java 集合深入理解系列,持續(xù)更新~ JVM 源碼分析之 System.currentTimeMillis 及 nanoTime 原理詳解 JVM 源碼分析之 System.currentTimeMi...

    Freeman 評論0 收藏0
  • Java集合干貨——ArrayList源碼分析

    摘要:關(guān)于的具體實(shí)現(xiàn),一些基本的都也知道,譬如數(shù)組實(shí)現(xiàn),線程不安全等等,但是更加具體的就很少去了解了,例如初始化的長度,擴(kuò)容等。 前言 在之前的文章中我們提到過ArrayList,ArrayList可以說是每一個(gè)學(xué)java的人使用最多最熟練的集合了,但是知其然不知其所以然。關(guān)于ArrayList的具體實(shí)現(xiàn),一些基本的都也知道,譬如數(shù)組實(shí)現(xiàn),線程不安全等等,但是更加具體的就很少去了解了,例如:...

    Render 評論0 收藏0
  • Java集合干貨——ArrayList源碼分析

    摘要:關(guān)于的具體實(shí)現(xiàn),一些基本的都也知道,譬如數(shù)組實(shí)現(xiàn),線程不安全等等,但是更加具體的就很少去了解了,例如初始化的長度,擴(kuò)容等。 前言 在之前的文章中我們提到過ArrayList,ArrayList可以說是每一個(gè)學(xué)java的人使用最多最熟練的集合了,但是知其然不知其所以然。關(guān)于ArrayList的具體實(shí)現(xiàn),一些基本的都也知道,譬如數(shù)組實(shí)現(xiàn),線程不安全等等,但是更加具體的就很少去了解了,例如:...

    wupengyu 評論0 收藏0
  • ConcurrentModificationException,iterator迭代問題[源碼分析]

    摘要:單線程的迭代過程中刪除集合元素以上代碼會出現(xiàn)如下異常從后往前看第行代碼我們在執(zhí)行代碼行時(shí)調(diào)用了這個(gè)是調(diào)用返回的對象這個(gè)對象的方法如下圖方法首先它會調(diào)用這個(gè)方法這個(gè)方法很簡單就是比較這兩個(gè)值是不是相等不相等就拋出異常如下圖這兩個(gè)值為什么會不相 單線程的Iterator迭代過程中刪除集合元素 public class TestIterator { public static voi...

    zhoutk 評論0 收藏0
  • ConcurrentModificationException,iterator迭代問題[源碼分析]

    摘要:單線程的迭代過程中刪除集合元素以上代碼會出現(xiàn)如下異常從后往前看第行代碼我們在執(zhí)行代碼行時(shí)調(diào)用了這個(gè)是調(diào)用返回的對象這個(gè)對象的方法如下圖方法首先它會調(diào)用這個(gè)方法這個(gè)方法很簡單就是比較這兩個(gè)值是不是相等不相等就拋出異常如下圖這兩個(gè)值為什么會不相 單線程的Iterator迭代過程中刪除集合元素 public class TestIterator { public static voi...

    aaron 評論0 收藏0

發(fā)表評論

0條評論

DevTalking

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<