一文講透為什么遍歷LinkedList要用增強(qiáng)型for循環(huán)
for循環(huán)和鏈表介紹
我們都知道java中有個增強(qiáng)型for循環(huán),這個for循環(huán)很方便,如果不需要知道當(dāng)前遍歷到第幾個的話可以跟普通for循環(huán)替換使用,也有人知道這倆好像有那么一點點不一樣,但為什么不一樣就不知道了。
我們還知道LinkedList是一個雙向鏈表,這個集合應(yīng)該是唯一一個既實現(xiàn)了List接口又實現(xiàn)了Queue接口的集合類。 鏈表這種數(shù)據(jù)結(jié)構(gòu),跟數(shù)組相比,優(yōu)勢在插入,劣勢在遍歷,那如果要遍歷一個鏈表,就要從頭開始遍歷,否則根本不知道下一個Node是什么。
增強(qiáng)for循環(huán)為什么遍歷LinkedList那么快
其實這個標(biāo)題不合適,應(yīng)該是為什么普通for循環(huán)遍歷LinkedList為什么那么慢。我們寫代碼驗證一下時間:
public void test() { LinkedList<Integer> list = new LinkedList<>(); for (int i = 0; i < 100000; i++) {//插入100000條數(shù)據(jù) list.add(i); } int index = 0;//記錄最后一個元素 long time1 = System.currentTimeMillis(); for (int i = 0; i < list.size(); i++) {//普通for循環(huán)遍歷 index = list.get(i); } long time2 = System.currentTimeMillis(); LogUtil.Companion.d("1:" + (time2 - time1) + " index->" + index); for (int i : list) {//增強(qiáng)for循環(huán)遍歷 index = i; } long time3 = System.currentTimeMillis(); LogUtil.Companion.d("2:" + (time3 - time2) + " index->" + index); Iterator<Integer> iterator = list.listIterator(); while (iterator.hasNext()) {//iterator遍歷 index = iterator.next(); } long time4 = System.currentTimeMillis(); LogUtil.Companion.d("3:" + (time4 - time3) + " index->" + index); }
運行結(jié)果:
1:5056 index->99999
2:12 index->99999
3:1 index->99999
其實增強(qiáng)型for循環(huán)底層就是用iterator
實現(xiàn)的,可以分析兩者的字節(jié)碼得出這個結(jié)論,這里我們不分析,算作一致結(jié)論。來看上面的結(jié)果,發(fā)現(xiàn)普通for循環(huán)遍歷的時間跟增強(qiáng)for循環(huán)和iterator相比簡直令人發(fā)指。 為什么會這樣呢?我們看LinkedList的源碼一探究竟。
LinkedList是一個雙向鏈表,用Head跟Tail兩個Node記錄了頭尾節(jié)點。
LinkedList相關(guān)源碼分析
普通for循環(huán)
我們看到其實普通for循環(huán)只是調(diào)用了LinkedList的 get(index) 方法:
//LinkedList.java public E get(int index) { checkElementIndex(index); //只是檢測是否數(shù)組越界 return node(index).item; //調(diào)用了node(index)方法 /** * Returns the (non-null) Node at the specified element index. */ Node<E> node(int index) { // assert isElementIndex(index); //判斷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; } } }
get方法很簡單,只是調(diào)用了node方法,node方法也很簡單,只是判斷了index是否是小于size/2,小于說明離Head近一點,否則說明離tail近,離哪個近就從哪一頭開始暴力遍歷,所以如果LinkedList有100000個Node,那最遠(yuǎn)的那個Node如果調(diào)用get方法就需要遍歷50000次。
所以普通for循環(huán)遍歷一次n個節(jié)點的LinkedList需要1+2+3+...+n/2+n/2+...+3+2+1
次,時間復(fù)雜度可以寫作O(n^2^)
。
增強(qiáng)for循環(huán)
可以看到最終是調(diào)用了LinkedList的內(nèi)部類ListItr
//LinkedList.java public ListIterator<E> listIterator(int index) { checkPositionIndex(index); return new ListItr(index); } private class ListItr implements ListIterator<E> { private Node<E> lastReturned; private Node<E> next; private int nextIndex; private int expectedModCount = modCount; ListItr(int index) { // assert isPositionIndex(index); next = (index == size) ? null : node(index); nextIndex = index; } public boolean hasNext() { return nextIndex < size; } public E next() { checkForComodification(); if (!hasNext()) throw new NoSuchElementException(); lastReturned = next; next = next.next; nextIndex++; return lastReturned.item; } ... final void checkForComodification() { if (modCount != expectedModCount) throw new ConcurrentModificationException(); } }
這里我們忽略一部分代碼先只看for循環(huán)涉及的方法,代碼其實也很簡單。如果 hasNext() 存在,就調(diào)用next() ,兩個Node:lastReturned和next。
每次獲取index的時候調(diào)用 ListItr(int index) 后next會指向當(dāng)前index的Node。
調(diào)用next的時候lastReturned會指向next也就是當(dāng)前index的Node,next指向next.next,所以每次遍歷的時候只要賦值一次就可以得到next的節(jié)點,所以遍歷一個n個節(jié)點的LinkedList就是需要n次。
所以用iterator遍歷的話時間復(fù)雜度就是O(n)。
這就是為什么兩個for循環(huán)的方式這么區(qū)別這么大了~我們也可以直觀的看出來當(dāng)n到達(dá)十萬這個級別的時候O(n^2^)和O(n)差別有多大了。
不知道各位發(fā)現(xiàn)沒有,Iterator里面每個操作都先調(diào)用了 checkForComodification() 方法,判斷 (modCount != expectedModCount) 是否相等。
各位應(yīng)該發(fā)現(xiàn)了ListItr有一個賦值,把modCount賦值給了expectedModCount,但每次調(diào)用遍歷或者addsetget的時候都會判斷這兩個值是否相等。
modCount是父類AbstractList的屬性,而每次調(diào)用add(),remove()方法的時候這個值都會變,也就是如果集合里面內(nèi)容修改了modCount都會發(fā)生改變。
So,在使用Iterator的時候不能調(diào)用add()或者remove()這些會改變集合內(nèi)容的方法。兩種情況:
- 在增強(qiáng)型for循環(huán)里面不能有add或者remove操作,使用Iterator迭代的時候不能做add或者remove操作。
- 如果有其他線程操作集合,需要加鎖避免改變集合,等待循環(huán)結(jié)束之后再修改。
否則都會報ConcurrentModificationException。
以上就是一文講透為什么遍歷LinkedList要用增強(qiáng)型for循環(huán)的詳細(xì)內(nèi)容,更多關(guān)于遍歷LinkedList增強(qiáng)型for循環(huán)的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
詳解如何配置Spring Batch批處理失敗重試機(jī)制
這篇文章主要來和大家一起探討一下如何在Spring批處理框架中配置重試邏輯,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-06-06SpringBoot中使用JeecgBoot的Autopoi導(dǎo)出Excel的方法步驟
這篇文章主要介紹了SpringBoot中使用JeecgBoot的Autopoi導(dǎo)出Excel的方法步驟,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09java.util.Date與java.sql.Date的區(qū)別
這篇文章主要介紹了java.util.Date與java.sql.Date的區(qū)別的相關(guān)資料,需要的朋友可以參考下2015-07-07Java的Flowable工作流之加簽轉(zhuǎn)簽詳解
這篇文章主要介紹了Java的Flowable工作流之加簽轉(zhuǎn)簽詳解,Flowable是一個開源的工作流引擎,它提供了一套強(qiáng)大的工具和功能,用于設(shè)計、執(zhí)行和管理各種類型的工作流程,需要的朋友可以參考下2023-11-11