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

Java中LinkedList真的是查找慢增刪快

 更新時(shí)間:2020年10月20日 10:50:50   作者:你在我家門口  
這篇文章主要介紹了Java中LinkedList真的是查找慢增刪快,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

測試結(jié)果

廢話不多說,先上測試結(jié)果。作者分別在ArrayList和LinkedList的頭部、尾部和中間三個(gè)位置插入與查找100000個(gè)元素所消耗的時(shí)間來進(jìn)行對比測試,下面是測試結(jié)果

(感謝@Hosalo的指正,在這里說明一下測試的環(huán)境,尾部插入是在空表的基礎(chǔ)上測試的,頭部和中間位置插入是在已存在100000個(gè)元素的表上進(jìn)行測試的)

插入 查找
ArrayList尾部 26ms 4ms
ArrayList頭部 2887ms 3ms
ArrayList中間 1936ms 4ms
LinkedList尾部 28ms 9ms
LinkedList頭部 15ms 11ms
LinkedList中間 12310ms 11387ms

測試結(jié)論

  • ArrayList的查找性能絕對是一流的,無論查詢的是哪個(gè)位置的元素
  • ArrayList除了尾部插入的性能較好外(位置越靠后性能越好),其他位置性能就不如人意了
  • LinkedList在頭尾查找、插入性能都是很棒的,但是在中間位置進(jìn)行操作的話,性能就差很遠(yuǎn)了,而且跟ArrayList完全不是一個(gè)量級的

源碼分析

我們把Java中的ArrayList和LinkedList就是分別對順序表和雙向鏈表的一種實(shí)現(xiàn),所以在進(jìn)行源碼分析之前,我們先來簡單回顧一下數(shù)據(jù)結(jié)構(gòu)中的順序表與雙向鏈表中的關(guān)鍵概念

  • 順序表:需要申請連續(xù)的內(nèi)存空間保存元素,可以通過內(nèi)存中的物理位置直接找到元素的邏輯位置。在順序表中間插入or刪除元素需要把該元素之后的所有元素向前or向后移動(dòng)。
  • 雙向鏈表:不需要申請連續(xù)的內(nèi)存空間保存元素,需要通過元素的頭尾指針找到前繼與后繼元素(查找元素的時(shí)候需要從頭or尾開始遍歷整個(gè)鏈表,直到找到目標(biāo)元素)。在雙向鏈表中插入or刪除元素不需要移動(dòng)元素,只需要改變相關(guān)元素的頭尾指針即可。

所以我們潛意識(shí)會(huì)認(rèn)為:ArrayList查找快,增刪慢。LinkedList查找慢,增刪快。但實(shí)際上真的是這樣的嗎?我們一起來看看吧。

測試程序

測試程序代碼基本沒有什么營養(yǎng),這里就不貼出來了,但是得把程序的運(yùn)行結(jié)果貼出來,方便逐個(gè)分析。

運(yùn)行結(jié)果

ArrayList尾部插入100000個(gè)元素耗時(shí):26ms
LinkedList尾部插入100000個(gè)元素耗時(shí):28ms
ArrayList頭部插入100000個(gè)元素耗時(shí):859ms
LinkedList頭部插入100000個(gè)元素耗時(shí):15ms
ArrayList中間插入100000個(gè)元素耗時(shí):1848ms
LinkedList中間插入100000個(gè)元素耗時(shí):15981ms
ArrayList頭部讀取100000個(gè)元素耗時(shí):7ms
LinkedList頭部讀取100000個(gè)元素耗時(shí):11ms
ArrayList尾部讀取100000個(gè)元素耗時(shí):12ms
LinkedList尾部讀取100000個(gè)元素耗時(shí):9ms
ArrayList中間讀取100000個(gè)元素耗時(shí):13ms
LinkedList中間讀取100000個(gè)元素耗時(shí):11387ms

ArrayList尾部插入

源碼

add(E e)方法
  public boolean add(E e) {
    // 檢查是否需要擴(kuò)容
    ensureCapacityInternal(size + 1); // Increments modCount!!
    // 直接在尾部添加元素
    elementData[size++] = e;
    return true;
  }

可以看出,對ArrayList的尾部插入,直接插入即可,無須額外的操作。

LinkedList尾部插入

源碼

LinkedList中定義了頭尾節(jié)點(diǎn)
  /**
   * Pointer to first node.
   */
  transient Node<E> first;

  /**
   * Pointer to last node.
   */
  transient Node<E> last;

add(E e)方法,該方法中調(diào)用了linkLast(E e)方法

  public boolean add(E e) {
    linkLast(e);
    return true;
  }

linkLast(E e)方法,可以看出,在尾部插入的時(shí)候,并不需要從頭開始遍歷整個(gè)鏈表,因?yàn)橐呀?jīng)事先保存了尾結(jié)點(diǎn),所以可以直接在尾結(jié)點(diǎn)后面插入元素

  /**
   * Links e as last element.
   */
  void linkLast(E e) {
    // 先把原來的尾結(jié)點(diǎn)保存下來
    final Node<E> l = last;
    // 創(chuàng)建一個(gè)新的結(jié)點(diǎn),其頭結(jié)點(diǎn)指向last
    final Node<E> newNode = new Node<>(l, e, null);
    // 尾結(jié)點(diǎn)置為newNode
    last = newNode;
    if (l == null)
      first = newNode;
    else
      // 修改原先的尾結(jié)點(diǎn)的尾結(jié)點(diǎn),使其指向新的尾結(jié)點(diǎn)
      l.next = newNode;
    size++;
    modCount++;
  }

總結(jié)

對于尾部插入而言,ArrayList與LinkedList的性能幾乎是一致的

ArrayList頭部插入

源碼

add(int index, E element)方法,可以看到通過調(diào)用系統(tǒng)的數(shù)組復(fù)制方法來實(shí)現(xiàn)了元素的移動(dòng)。所以,插入的位置越靠前,需要移動(dòng)的元素就會(huì)越多

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

    ensureCapacityInternal(size + 1); // Increments modCount!!
    // 把原來數(shù)組中的index位置開始的元素全部復(fù)制到index+1開始的位置(其實(shí)就是index后面的元素向后移動(dòng)一位)
    System.arraycopy(elementData, index, elementData, index + 1,
             size - index);
    // 插入元素
    elementData[index] = element;
    size++;
  }

LinkedList頭部插入

源碼

add(int index, E element)方法,該方法先判斷是否是在尾部插入,如果是調(diào)用linkLast()方法,否則調(diào)用linkBefore(),那么是否真的就是需要重頭開始遍歷呢?我們一起來看看

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

    if (index == size)
      linkLast(element);
    else
      linkBefore(element, node(index));
  }

在頭尾以外的位置插入元素當(dāng)然得找出這個(gè)位置在哪里,這里面的node()方法就是關(guān)鍵所在,這個(gè)函數(shù)的作用就是根據(jù)索引查找元素,但是它會(huì)先判斷index的位置,如果index比size的一半(size >> 1,右移運(yùn)算,相當(dāng)于除以2)要小,就從頭開始遍歷。否則,從尾部開始遍歷。從而可以知道,對于LinkedList來說,操作的元素的位置越往中間靠攏,效率就越低

  Node<E> node(int index) {
    // assert isElementIndex(index);

    if (index < (size >> 1)) {
      Node<E> x = first;
      for (int i = 0; i < index; i++)
        x = x.next;
      return x;
    } else {
      Node<E> x = last;
      for (int i = size - 1; i > index; i--)
        x = x.prev;
      return x;
    }
  }

這個(gè)函數(shù)的工作就只是負(fù)責(zé)把元素插入到相應(yīng)的位置而已,關(guān)鍵的工作在node()方法中已經(jīng)完成了

  void linkBefore(E e, Node<E> succ) {
    // assert succ != null;
    final Node<E> pred = succ.prev;
    final Node<E> newNode = new Node<>(pred, e, succ);
    succ.prev = newNode;
    if (pred == null)
      first = newNode;
    else
      pred.next = newNode;
    size++;
    modCount++;
  }

總結(jié)

  • 對于LinkedList來說,頭部插入和尾部插入時(shí)間復(fù)雜度都是O(1)
  • 但是對于ArrayList來說,頭部的每一次插入都需要移動(dòng)size-1個(gè)元素,效率可想而知
  • 但是如果都是在最中間的位置插入的話,ArrayList速度比LinkedList的速度快將近10倍

ArrayList、LinkedList查找

  • 這就沒啥好說的了,對于ArrayList,無論什么位置,都是直接通過索引定位到元素,時(shí)間復(fù)雜度O(1)
  • 而對于LinkedList查找,其核心方法就是上面所說的node()方法,所以頭尾查找速度極快,越往中間靠攏效率越低

到此這篇關(guān)于Java中LinkedList真的是查找慢增刪快 的文章就介紹到這了,更多相關(guān)Java LinkedList查找慢增刪快 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家! 

相關(guān)文章

  • 實(shí)例講解Java并發(fā)編程之變量

    實(shí)例講解Java并發(fā)編程之變量

    這篇文章主要介紹了實(shí)例講解Java并發(fā)編程之變量,本文講解了編寫線程安全需要關(guān)心的共享變量和可變變量,需要的朋友可以參考下
    2015-04-04
  • Java 實(shí)戰(zhàn)項(xiàng)目錘煉之網(wǎng)上花店商城的實(shí)現(xiàn)流程

    Java 實(shí)戰(zhàn)項(xiàng)目錘煉之網(wǎng)上花店商城的實(shí)現(xiàn)流程

    讀萬卷書不如行萬里路,只學(xué)書上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實(shí)戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+jsp+servlet+mysql+ajax實(shí)現(xiàn)一個(gè)網(wǎng)上花店商城系統(tǒng),大家可以在過程中查缺補(bǔ)漏,提升水平
    2021-11-11
  • 詳解JAVA常用的時(shí)間操作【實(shí)用】

    詳解JAVA常用的時(shí)間操作【實(shí)用】

    本文主要介紹了JAVA一些常用的時(shí)間操作,很實(shí)用,相信大家在開發(fā)項(xiàng)目時(shí)會(huì)用到,下面就跟小編一起來看下吧
    2016-12-12
  • Java instanceof關(guān)鍵字用法詳解及注意事項(xiàng)

    Java instanceof關(guān)鍵字用法詳解及注意事項(xiàng)

    instanceof 是 Java 的保留關(guān)鍵字。它的作用是測試它左邊的對象是否是它右邊的類的實(shí)例,返回 boolean 的數(shù)據(jù)類型。本文重點(diǎn)給大家介紹Java instanceof關(guān)鍵字用法詳解及注意事項(xiàng),需要的朋友參考下吧
    2021-09-09
  • 解決spring security中遇到的問題

    解決spring security中遇到的問題

    這篇文章主要介紹了解決spring security中遇到的問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-01-01
  • MyBatis-Plus攔截器對敏感數(shù)據(jù)實(shí)現(xiàn)加密

    MyBatis-Plus攔截器對敏感數(shù)據(jù)實(shí)現(xiàn)加密

    做課程項(xiàng)目petstore時(shí)遇到需要加密屬性的問題,而MyBatis-Plus為開發(fā)者提供了攔截器的相關(guān)接口,本文主要介紹通過MyBatis-Plus的攔截器接口自定義一個(gè)攔截器類實(shí)現(xiàn)敏感數(shù)據(jù)如用戶密碼的加密功能,感興趣的可以了解一下
    2021-11-11
  • java求解漢諾塔問題示例

    java求解漢諾塔問題示例

    漢諾塔問題的描述如下:有3根柱子A、B和C,在A上從上往下按照從小到大的順序放著一些圓盤,以B為中介,把盤子全部移動(dòng)到C上。移動(dòng)過程中,要求任意盤子的下面要么沒有盤子,要么只能有比它大的盤子。編程實(shí)現(xiàn)3階漢諾塔的求解步驟
    2014-02-02
  • 一文搞清楚Spring事務(wù)

    一文搞清楚Spring事務(wù)

    Spring事務(wù)是指在Spring框架中對于數(shù)據(jù)庫操作的一種支持,它通過對一組數(shù)據(jù)庫操作進(jìn)行整體控制來保證數(shù)據(jù)的一致性和完整性。本文介紹Spring事務(wù)介紹的非常詳細(xì),有需要的朋友可以參考本文
    2023-04-04
  • java內(nèi)部類原理與用法詳解

    java內(nèi)部類原理與用法詳解

    這篇文章主要介紹了java內(nèi)部類原理與用法,結(jié)合實(shí)例形式分析了Java內(nèi)部類的概念、原理、分類及相關(guān)使用技巧,需要的朋友可以參考下
    2019-05-05
  • Java線程代碼的實(shí)現(xiàn)方法

    Java線程代碼的實(shí)現(xiàn)方法

    下面小編就為大家?guī)硪黄狫ava線程代碼的實(shí)現(xiàn)方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-08-08

最新評論