01_ArrayList核心源碼剖析

一拦耐、基本原理

  1. 數(shù)組的長度是固定的弯蚜,java里面數(shù)組都是定長數(shù)組十籍,如果不停的往ArrayList里面塞入這個數(shù)據(jù)蛆封,此時元素?cái)?shù)量超過了初始大小,此時就會發(fā)生一個數(shù)組的擴(kuò)容勾栗,就會搞一個更大的數(shù)組惨篱,把以前的數(shù)組拷貝到新的數(shù)組里面去
  2. 缺點(diǎn)一、這個數(shù)組擴(kuò)容+元素拷貝的過程围俘,相對來說會慢一些. 所以砸讳,不要頻繁的往arralist里面去塞數(shù)據(jù)琢融,導(dǎo)致他頻繁的數(shù)組擴(kuò)容,避免擴(kuò)容的時候較差的性能影響了系統(tǒng)的運(yùn)行
  3. 缺點(diǎn)二簿寂、數(shù)組來實(shí)現(xiàn)漾抬,數(shù)組你要是往數(shù)組的中間加一個元素,是不是要把數(shù)組中那個新增元素后面的元素全部往后面挪動一位常遂,所以說纳令,如果你是往ArrayList中間插入一個元素,性能比較差克胳,會導(dǎo)致他后面的大量的元素挪動一個位置
  4. 優(yōu)點(diǎn)一平绩、基于數(shù)組來實(shí)現(xiàn),非常適合隨機(jī)讀漠另,你可以隨機(jī)的去讀數(shù)組中的某個元素捏雌,list.get(10),相當(dāng)于是在獲取第11個元素笆搓,這個隨機(jī)讀的性能是比較高的腹忽,隨機(jī)讀,list.get(2)砚作,list.get(20)窘奏,隨機(jī)讀list里任何一個元素
  5. 為什么基于數(shù)組的ArrayList隨機(jī)讀寫會很快,而插入性能較低葫录,因?yàn)閿?shù)組是基于內(nèi)存里連續(xù)的空間着裹,只要根據(jù)Index+offset就可以計(jì)算出我們想訪問的那個元素在內(nèi)存中的地址

源碼

代碼片段一、

public static void main(String[] args) {
    List<String> list = new ArrayList<String>();
    // 代碼片段二米同、
    list.add("zhangsan");
    list.add("lisi");
    list.add("wuwang");
    System.out.println(list.toString());
    System.out.println(list.toString());
    // 代碼片段六骇扇、
    list.get(2);
    // 代碼片段七、這里其實(shí)是向集合中的index為1的地方面粮,插入一個新的元素
    list.add(1,"zhaoliu");
}

代碼片段二少孝、

/**
 * Appends the specified element to the end of this list.
 *
 * @param e element to be appended to this list
 * @return <tt>true</tt> (as specified by {@link Collection#add})
 */
public boolean add(E e) {
    // 這里是在計(jì)算當(dāng)前元素應(yīng)該放在list中的index索引,代碼片段三熬苍、
    // 這里的size其實(shí)就是當(dāng)前集合的大小稍走,這里要和capacity區(qū)分開
    ensureCapacityInternal(size + 1);  // Increments modCount!!
    // 將當(dāng)前元素放入到集合中,index=size++
    elementData[size++] = e;
    return true;
}

代碼片段三柴底、

private void ensureCapacityInternal(int minCapacity) {
    // 代碼片段四婿脸、
    ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}

// 通過代碼片段四,第一次進(jìn)來柄驻,minCapacity的值為10
private void ensureExplicitCapacity(int minCapacity) {
    modCount++;


    // overflow-conscious code
    // 如果minCapacity大于集合現(xiàn)在的大小的話狐树,就會走k擴(kuò)容邏輯
    if (minCapacity - elementData.length > 0)
        grow(minCapacity);
}

代碼片段四、

/**
* elementData:代表當(dāng)前集合的底層數(shù)組
* minCapacity:當(dāng)前數(shù)組的大小+1
* 所以第一次調(diào)用list的add()方法的時候鸿脓,elementData是空的抑钟,minCapacity為1
* 返回DEFAULT_CAPACITY涯曲,集合的初始化大小 10
*/
private static int calculateCapacity(Object[] elementData, int minCapacity) {
    if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        return Math.max(DEFAULT_CAPACITY, minCapacity);
    }
    return minCapacity;
}

代碼片段五、

/**
 * 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
    // 集合現(xiàn)在的大小
    int oldCapacity = elementData.length;
    // 擴(kuò)容邏輯就是:oldCapacity(老的大小) + oldCapacity/ 2
    // 比如原來大小10在塔,現(xiàn)在擴(kuò)容掀抹,就是10 + 10/2 = 15
    int newCapacity = oldCapacity + (oldCapacity >> 1);
    // 如果新的倉位15大于minCapacity
    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:
    // 這里進(jìn)行數(shù)組拷貝,完成老數(shù)組到新數(shù)組的拷貝
    elementData = Arrays.copyOf(elementData, newCapacity);
}

代碼片段六心俗、

/**
 * Returns the element at the specified position in this list.
 *
 * @param  index index of the element to return
 * @return the element at the specified position in this list
 * @throws IndexOutOfBoundsException {@inheritDoc}
 */
public E get(int index) {
    // 如下面代碼,如果index超過了size蓉驹,就會報出來一個數(shù)組越界異常
    rangeCheck(index);

    // 返回index下的元素城榛,所以,集合的讀是很快的
    return elementData(index);
}

/**
 * Checks if the given index is in range.  If not, throws an appropriate
 * runtime exception.  This method does *not* check if the index is
 * negative: It is always used immediately prior to an array access,
 * which throws an ArrayIndexOutOfBoundsException if index is negative.
 */
private void rangeCheck(int index) {
    if (index >= size)
        throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
}

代碼片段七态兴、

/**
* 在指定位置插入一個新的元素狠持,然后指定位置以后的元素進(jìn)行重新排序
 * Inserts the specified element at the specified position in this
 * list. Shifts the element currently at that position (if any) and
 * any subsequent elements to the right (adds one to their indices).
 *
 * @param index index at which the specified element is to be inserted
 * @param element element to be inserted
 * @throws IndexOutOfBoundsException {@inheritDoc}
 */
public void add(int index, E element) {
     // 檢查index是否越界,越界則拋出角標(biāo)越界異常
    rangeCheckForAdd(index);

    // 代碼片段四中的檢查是否需要擴(kuò)容
    ensureCapacityInternal(size + 1);  // Increments modCount!!
    // 這里的數(shù)組拷貝瞻润,然后index后面的元素向后移動
    System.arraycopy(elementData, index, elementData, index + 1,
                     size - index);
    // 將新的元素放入到指定index中
    elementData[index] = element;
    size++;
}

/**
 * A version of rangeCheck used by add and addAll.
 */
private void rangeCheckForAdd(int index) {
    // 檢查index是否越界
    if (index > size || index < 0)
        throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
}

三喘垂、總結(jié)

  1. 其實(shí)對于大部分Java的程序員來說,ArrayList的基本原理都沒啥問題绍撞,但是我們?yōu)槭裁催€要繼續(xù)分析ArrayList的源碼呢正勒,主要是,隨著我們工作年限的增加傻铣,在去面試的話章贞,基本上都是往死里問,所以對于基礎(chǔ)的集合非洲,我們還是很有必要來分析一下他的源碼
  2. 我們僅僅是研究一下核心源碼鸭限,如果對ArrzyList中1500行左右的源碼全部剖析的話,首先我們肯定記不住這么多两踏,其次還可能忽略核心功能败京,如果對于核心功能已經(jīng)了解的話,其他的API其實(shí)都不難的
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末梦染,一起剝皮案震驚了整個濱河市赡麦,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌帕识,老刑警劉巖隧甚,帶你破解...
    沈念sama閱讀 216,651評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異渡冻,居然都是意外死亡戚扳,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,468評論 3 392
  • 文/潘曉璐 我一進(jìn)店門族吻,熙熙樓的掌柜王于貴愁眉苦臉地迎上來帽借,“玉大人珠增,你說我怎么就攤上這事】嘲” “怎么了蒂教?”我有些...
    開封第一講書人閱讀 162,931評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長脆荷。 經(jīng)常有香客問我凝垛,道長,這世上最難降的妖魔是什么蜓谋? 我笑而不...
    開封第一講書人閱讀 58,218評論 1 292
  • 正文 為了忘掉前任梦皮,我火速辦了婚禮,結(jié)果婚禮上桃焕,老公的妹妹穿的比我還像新娘剑肯。我一直安慰自己,他們只是感情好观堂,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,234評論 6 388
  • 文/花漫 我一把揭開白布让网。 她就那樣靜靜地躺著,像睡著了一般师痕。 火紅的嫁衣襯著肌膚如雪溃睹。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,198評論 1 299
  • 那天胰坟,我揣著相機(jī)與錄音丸凭,去河邊找鬼。 笑死腕铸,一個胖子當(dāng)著我的面吹牛惜犀,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播狠裹,決...
    沈念sama閱讀 40,084評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼虽界,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了涛菠?” 一聲冷哼從身側(cè)響起莉御,我...
    開封第一講書人閱讀 38,926評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎俗冻,沒想到半個月后礁叔,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,341評論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡迄薄,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,563評論 2 333
  • 正文 我和宋清朗相戀三年琅关,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片讥蔽。...
    茶點(diǎn)故事閱讀 39,731評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡涣易,死狀恐怖画机,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情新症,我是刑警寧澤步氏,帶...
    沈念sama閱讀 35,430評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站徒爹,受9級特大地震影響荚醒,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜隆嗅,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,036評論 3 326
  • 文/蒙蒙 一界阁、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧榛瓮,春花似錦、人聲如沸巫击。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,676評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽坝锰。三九已至粹懒,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間顷级,已是汗流浹背凫乖。 一陣腳步聲響...
    開封第一講書人閱讀 32,829評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留弓颈,地道東北人帽芽。 一個月前我還...
    沈念sama閱讀 47,743評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像翔冀,于是被迫代替她去往敵國和親导街。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,629評論 2 354

推薦閱讀更多精彩內(nèi)容

  • 一纤子、基本方法解析 1搬瑰、構(gòu)造器 ArrayList有三種創(chuàng)建方式 ArrayList的底層實(shí)現(xiàn)方式為數(shù)組,在創(chuàng)建Ar...
    慕凌峰閱讀 611評論 0 3
  • 概要 上一章控硼,我們學(xué)習(xí)了Collection的架構(gòu)泽论。這一章開始,我們對Collection的具體實(shí)現(xiàn)類進(jìn)行講解卡乾;首...
    廢棄的root閱讀 892評論 0 4
  • 一.線性表 定義:零個或者多個元素的有限序列翼悴。也就是說它得滿足以下幾個條件:??①該序列的數(shù)據(jù)元素是有限的。??②...
    Geeks_Liu閱讀 2,697評論 1 12
  • 以下源碼基于 Android SDK 23幔妨, 與JDK中略有差別抄瓦,但基本相同潮瓶;整體源碼由 構(gòu)造、添加(add)钙姊、設(shè)...
    Jacky_Y閱讀 442評論 0 3
  • 久違的晴天毯辅,家長會。 家長大會開好到教室時煞额,離放學(xué)已經(jīng)沒多少時間了思恐。班主任說已經(jīng)安排了三個家長分享經(jīng)驗(yàn)。 放學(xué)鈴聲...
    飄雪兒5閱讀 7,522評論 16 22