集合(或者叫容器)是Java的核心知識點,它有着很深的深度。咱們這裏不會設計多深,僅僅做爲了解入門,深刻了解請移步各類集合源碼文章。好的,下面正是開始介紹...java
咱們知道,Java是一門面向對象編程語言,這也就意味着程序中存在着大量的對象。這個時候問題就來了,咱們如何很好的存放和操做對象呢?若是你能明白這個問題,那麼你就知道了「集合爲什麼而生」這個問題的答案。面試
總結一句: Java給咱們提供了工具(集合)方便咱們去存放和操做多個Java對象編程
Java集合的目的是方便操做多個對象,同時,它提供一系列的API來供咱們操做。所以,在初學Java集合的時候咱們更多的是學習這些API的用法。設計模式
對Java集合的API使用有必定了解以後,咱們就應該從面向對象的角度去理解它。爲何會抽象出多個接口,以及每一個接口有什麼特徵。數組
咱們能夠總結出幾個經常使用的實現類,這幾個經常使用的實現類咱們必須知道它的數據結構是什麼,何時使用這個類。微信
同時,你還須要學習和了解數據結構:數據結構
學完上面的內容後,咱們基本掌握了經常使用集合的數據結構,也就知道了如何選擇合適的集合容器去存儲咱們的對象。總之,學完常見實現類的數據結構以後,對它們的使用場景也有了更加清晰的認識。框架
Java集合是一個很龐大的知識點,話很少說,上圖感覺下:編程語言
Java容器可分爲兩大類:工具
迭代器是一種設計模式,它是一個對象,它能夠遍歷並選擇序列中的對象,而開發人員不須要了解該序列的底層結構。迭代器一般被稱爲「輕量級」對象,由於建立它的代價小
咱們能夠發現一個特色,上述全部的集合類,除了Map系列的集合,Collection集合都實現了Iterator接口。咱們能夠在源碼中追溯到集合的頂層接口,好比Collection接口,能夠看到它繼承的是類Iterable。
它是Java集合的頂層接口(不包括Map系列的集合,Map接口是Map系列集合的頂層接口)
因此除了Map系列的集合,我麼都能經過迭代器來對集合中的元素進行遍歷。
Map集合的子類能夠用keyset()方法轉換成Set集合遍歷
Iterable迭代器一共4個方法:
hasNext():判斷下個迭代器是否還有下一個元素
next():返回下一個元素的值,而且把自身offset移動下一位
remove():這個能夠刪除用這個迭代器集合中的元素(注意若是刪除以後仍是前面得到的迭代器,你會發現原來的迭代器仍是沒變,得從新得到刪除元素以後的迭代器)
forEachRemaining:1.8的新方法 能夠直接遍歷迭代器剩下的元素,若是從最開始的話就是遍歷全部的迭代器
補充: 還有一個ListIterator接口,它繼承了Iterator接口,但只能用於List集合。它是Iterator接口的升級版,裏面除了Iterator含有的功能外,還具備一些其餘的功能。Iterator遍歷集合元素時,只能單向遍歷,而ListIterator能夠雙向進行遍歷、添加元素、設置元素
當使用Iterator對集合元素進行迭代時,Iterator並非把集合元素自己傳給了迭代變量,而是把集合元素的值傳給了迭代變量(就如同參數傳遞是值傳遞,基本數據類型傳遞的是值,引用類型傳遞的僅僅是對象的引用變量),因此修改迭代變量的值對集合元素自己沒有任何影響。
下面的程序演示了這一點:
public class IteratorExample { public static void main(String[] args){ List<String> list =Arrays.asList("java語言","C語言","C++語言"); Iterator<String> iterator = list.iterator(); while(iterator.hasNext()){ String next = iterator.next();//集合元素的值傳給了迭代變量,僅僅傳遞了對象引用。保存的僅僅是指向對象內存空間的地址 next ="修改後的"; System.out.println(next); } System.out.println(list); } }
輸出結果:
修改後的 修改後的 修改後的 [java語言, C語言, C++語言]
Collection的做用就是規定了一個集合有哪些基本的操做。 這裏主要是插入數據,清空數據,是否包含,是否相等,集合裏的數據個數和轉化成熟組這幾種操做。
接口中定義的方法:
Collection接口是Set、List、Queue的父接口。下面逐一介紹:
Set集合與Collection集合基本相同,沒有提供任何額外的方法。實際上Set就是Collection,只是行爲略有不一樣(Set不容許包含重複元素)。
Set集合不容許包含相同的元素,若是試圖把兩個相同的元素加入同一個Set集合中,則添加操做失敗,add()方法返回false,且新元素不會被加入。
List集合表明一個元素有序、可重複的集合,集合中每一個元素都有其對應的順序索引。List集合容許使用重複元素,能夠經過索引來訪問指定位置的集合元素 。List集合默認按元素的添加順序設置元素的索引,例如第一個添加的元素索引爲0,第二個添加的元素索引爲1......
List做爲Collection接口的子接口,可使用Collection接口裏的所有方法。並且因爲List是有序集合,所以List集合裏增長了一些根據索引來操做集合元素的方法:
除此以外,Java 8還爲List接口添加了以下兩個默認方法:
Queue用戶模擬隊列這種數據結構,隊列一般是指「先進先出」(FIFO,first-in-first-out)的容器。隊列的頭部是在隊列中存放時間最長的元素,隊列的尾部是保存在隊列中存放時間最短的元素。新元素插入(offer)到隊列的尾部,訪問元素(poll)操做會返回隊列頭部的元素。一般,隊列不容許隨機訪問隊列中的元素。
接口中定義的方法:
Map用戶保存具備映射關係的數據,所以Map集合裏保存着兩組數,一組值用戶保存Map裏的key,另外一組值用戶保存Map裏的value,key和value均可以是任何引用類型的數據。Map的key不容許重複,即同一個Map對象的任何兩個key經過equals方法比較老是返回false。
以下圖所描述,key和value之間存在單向一對一關係,即經過指定的key,總能找到惟一的、肯定的value。從Map中取出數據時,只要給出指定的key,就能夠取出對應的value。
Map集合包括Map接口以及Map接口的全部實現類。Map集合具備如下特色:
Map接口經常使用的實現類有:HashMap、HashTable、TreeMap。
咱們先看下以下示例瞭解下Map集合的用法:
package collection; import java.util.*; public class Muster { public static void main(String[] args) { Map<Integer, String> platformMap = new HashMap<>(); platformMap.put(1, "博客園"); platformMap.put(2, "掘金"); platformMap.put(3, "微信公衆號"); platformMap.put(4, "我的博客"); // 嘗試添加劇復Map platformMap.put(4, "我的博客"); // 獲取全部的key Set<Integer> keys = platformMap.keySet(); for (Integer integer : keys) { System.out.println("Key:" + integer + ",Value:" + platformMap.get(integer)); } } }
以上代碼的輸出結果爲:
Key:1,Value:博客園 Key:2,Value:掘金 Key:3,Value:微信公衆號 Key:4,Value:我的博客
從日誌能夠看出,當咱們嘗試重加劇復Map時,並無添加成功。
關於Map集合的詳細用法,HashMap、HashTable、TreeMap的區別(這裏是重點,面試可能問的比較多,這裏不展開說明)
1. 與Set集合的關係
若是把Map裏的全部key放在一塊兒看,它們就組成了一個Set集合(全部的key沒有順序,key與key之間不能重複),實際上Map確實包含了一個keySet()
方法,用戶返回Map裏全部key組成的Set集合。
2. 與List集合的關係
若是把Map裏的全部value放在一塊兒來看,它們又很是相似於一個List:元素與元素之間能夠重複,每一個元素能夠根據索引來查找,只是Map中索引再也不使用整數值,而是以另一個對象做爲索引。
接口中定義的方法:
Map中還包括一個內部類Entry,該類封裝了一個key-value對。Entry包含以下三個方法:
Map集合最典型的用法就是成對地添加、刪除key-value對,而後就是判斷該Map中是否包含指定key,是否包含指定value,也能夠經過Map提供的keySet()
方法獲取全部key組成的集合,而後使用foreach
循環來遍歷Map的全部key,根據key便可遍歷全部的value。下面程序代碼示範Map的一些基本功能:
public class MapTest { public static void main(String[] args){ Day day1 = new Day(1, 2, 3); Day day2 = new Day(2, 3, 4); Map<String,Day> map = new HashMap<String,Day>(); //成對放入key-value對 map.put("第一個", day1); map.put("第二個", day2); //判斷是否包含指定的key System.out.println(map.containsKey("第一個")); //判斷是否包含指定的value System.out.println(map.containsValue(day1)); //循環遍歷 //1.得到Map中全部key組成的set集合 Set<String> keySet = map.keySet(); //2.使用foreach進行遍歷 for (String key : keySet) { //根據key得到指定的value System.out.println(map.get(key)); } //根據key來移除key-value對 map.remove("第一個"); System.out.println(map); } }
輸出結果:
true true Day [hour=2, minute=3, second=4] Day [hour=1, minute=2, second=3] {第二個=Day [hour=2, minute=3, second=4]}
參考: