首頁 > 軟體

Java迭代器與Collection介面超詳細講解

2022-07-19 14:02:44

關於迭代器你都知道什麼?

什麼是迭代器?

  所謂迭代的意思就是交換替代,迭代器並不是一種資料結構或者集合,而是可以過迭代器中的方法逐個存取集合中的每個元素的一種方法。提到迭代器最重要的就是Iterator介面,所有想要使用迭代器迭代的結構都需要實現或者繼承這個介面

迭代器的4個API

  Iterator介面包含4個方法,分別是next、hasNext、remove、forEachRemaining,接下來在學習構造器如何使用之前我們先學習一下它的四種方法

  首先一開始構造器中的有一個類似於指標的標誌,指在集合中第一個元素的前面,呼叫構器中的next方法會使這個"指標"的位置後移到第一和二個元素之間,然後返回它跨過的那個元素給構造器,也就是第一個元素,如果"指標"到最後一個元素後面再呼叫next方法的話就會導致丟擲NoSuchElementException

  remove方法則是刪除當前元素,可以理解為next方法返回的那個元素,如果呼叫remove方法之前沒有呼叫next方法或者是使用remove方法刪除該元素之後都會導致構造器為空,此時呼叫remove方法就會導致程式丟擲IllegalStateExceptions異常

  hasNext方法就是判斷迭代器是否還有可迭代的下一個元素,如果有的話就返回true否則返回false,為了避免上述兩種異常,remove方法每次都要配合另外兩種方法一起使用,每次呼叫next方法之前都使用hasNext方法判斷一下,避免沒有可迭代元素導致的異常,每一次呼叫remove之前都使用next方法獲得元素,避免迭代器為空導致的狀態異常

  forEachRemaining方法則可以直接遍歷迭代器中的每一個元素並呼叫方法引數中的Lambda表示式,直到迭代器將集合的元素全部迭代完為止

如何使用迭代器?

  接下來將結合上面的描述給出一個將集合中的元素全部刪除的標準程式碼,然後再給出兩個錯誤的程式碼,大家可以在使用迭代器的時候規避一下

正確寫法

// 建立一個ArrayList集合
Collection<String> strings = new ArrayList<>();
strings.add("a");
strings.add("b");
strings.add("c");
strings.add("d");
// 獲得迭代器物件
Iterator<String> iterator = strings.iterator();
// 使用while迴圈迭代集合中元素 使用hasNext方法判斷
while (iterator.hasNext()) {
    // 使用next獲取下一個元素
    iterator.next();
    // 刪除這個元素
    iterator.remove();
    // 列印集合中元素
    System.out.println(strings);
}

錯誤示範一

// 建立一個ArrayList集合
Collection<String> strings = new ArrayList<>();
strings.add("a");
strings.add("b");
strings.add("c");
strings.add("d");
// 獲得迭代器
Iterator<String> iterator = strings.iterator();
// 使用while迴圈迭代集合中元素 使用next方法判斷下一個元素是否為空
while (iterator.next() != null) {
    // 輸出這個元素
    System.out.println(iterator.next());
}

  第一個的錯誤原因:使用next方法判斷下一個元素是否為空,這樣就會導致"指標"到最後一個元素後面依舊會執行next方法,這樣就會導致丟擲NoSuchElementException異常;而且next方法每呼叫一次都會將"指標"向後移動一位,哪怕只是用於if判斷。所以程式就會每隔一個元素輸出一次,最後丟擲NoSuchElementException異常

錯誤示範二

// 建立一個ArrayList集合
Collection<String> strings = new ArrayList<>();
strings.add("a");
strings.add("b");
strings.add("c");
strings.add("d");
// 獲取迭代器物件 並使用while迴圈迭代集合中元素 使用hasNext方法判斷
while (strings.iterator().hasNext()) {
    // 獲取迭代器物件 並使用next獲取下一個元素 然後輸出
    System.out.println(strings.iterator().next());
}

  第二個的錯誤原因:每使用iterator方法獲得一次集合對應的迭代器物件,都會預設將"指標"放到第一個元素的前面,於是第二個錯誤示範中一直使用集合中的第一個元素"a"進行判斷有沒有下一個元素,所以會導致程式陷入死迴圈,迴圈體裡也會一直建立結合的迭代器物件,並將"指標"放到第一個元素的前面,然後呼叫next方法輸出元素"a"

Collection集合介面知多少?

  集合大體上可以分為兩種,一種是單列的Collection集合,一種就是雙列的Map集合,所謂的單雙列可以理解為元素中資料的個數,單列集合一個資料作為元素儲存,雙列集合兩個具有對映關係的資料作為元素儲存。這一篇我們先學習Collection集合介面的內容,Collection集合按照元素儲存是否有序又可分為List集合、Set集合

  前面之所以先學習Iterator介面的原因就是,Collection介面繼承了Iteator介面,於是它的子介面set和list都可以使用迭代器對集合中的元素進行迭代

為什麼不使用陣列而是集合?

  在學習集合之前,我們將儲存多個物件或者元素的任務都交給了陣列,但是陣列儲存元素有以下缺點:

①陣列一旦初始化之後,長度就確定不可修改,元素個數超出陣列長度的話會拋異常。

②陣列中提供的API很少,增加元素需要現將索引後的元素後移空出位置將元素新增進去,刪除元素需要刪除之後將後面的元素前移將空出來的位置補足,以上操作只能通過程式碼實現並沒有現成的API可以使用。

③陣列儲存資料的特點:有序、可重複,對於一些無需、不可重複的業務需求就很難滿足

  集合就可以很好的彌補陣列的上述缺點,而且集合提供了一組較為完善的資料結構,我們可以根據具體的業務需求來選擇具體使用的集合型別。比如說儲存元素無序不可重複的Set集合,有序可重複的List集合,具有對映關係的Map集合等

Collection介面的API都有什麼?

AbstractCollection類知多少?

  以上給出的Collection介面中的API都是抽象方法,也就意味著每一個此介面的實現類都需要重寫這些抽象方法,實際上Collection介面的直接或者間接實現類有很多,如果每一個都需要重寫這些方法的話就會很是麻煩。於是Java類庫的設計者提供了AbstractCollection類,該類中只將size方法和Iterator方法宣告為抽象,其他方法都提供了預設實現,如果子類不提供這些方法的方法體的話就使用該類中的預設實現

  下圖可知,list、set集合的實現類都直接或者間接的繼承了AbstractCollection類,為的就是簡化重寫Collection介面的抽象方法

到此這篇關於Java迭代器與Collection介面超詳細講解的文章就介紹到這了,更多相關Java迭代器與Collection介面內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com