面試必備:30個Java集合面試問題及答案

圖片描述
Java集合框架爲Java編程語言的基礎,也是Java面試中很重要的一個知識點。這裏,我列出了一些關於Java集合的重要問題和答案。java

1.Java集合框架是什麼?說出一些集合框架的優勢?面試

每種編程語言中都有集合,最初的Java版本包含幾種集合類:Vector、Stack、HashTable和Array。算法

隨着集合的普遍使用,Java1.2提出了囊括全部集合接口、實現和算法的集合框架。在保證線程安全的狀況下使用泛型和併發集合類,Java已經經歷了好久。它還包括在Java併發包中,阻塞接口以及它們的實現。編程

集合框架的部分優勢以下:數組

使用核心集合類下降開發成本,而非實現咱們本身的集合類。安全

隨着使用通過嚴格測試的集合框架類,代碼質量會獲得提升。數據結構

經過使用JDK附帶的集合類,能夠下降代碼維護成本。多線程

複用性和可操做性。併發

2.集合框架中的泛型有什麼優勢?框架

Java1.5引入了泛型,全部的集合接口和實現都大量地使用它。

泛型容許咱們爲集合提供一個能夠容納的對象類型,所以,若是你添加其它類型的任何元素,它會在編譯時報錯。

這避免了在運行時出現ClassCastException,由於你將會在編譯時獲得報錯信息。

泛型也使得代碼整潔,咱們不須要使用顯式轉換和instanceOf操做符。

它也給運行時帶來好處,由於不會產生類型檢查的字節碼指令。

3.Java集合框架的基礎接口有哪些?

Collection爲集合層級的根接口。一個集合表明一組對象,這些對象即爲它的元素。Java平臺不提供這個接口任何直接的實現。

Set是一個不能包含重複元素的集合。這個接口對數學集合抽象進行建模,被用來表明集合,就如一副牌。

List是一個有序集合,能夠包含重複元素。你能夠經過它的索引來訪問任何元素。List更像長度動態變換的數組。

Map是一個將key映射到value的對象.一個Map不能包含重複的key:每一個key最多隻能映射一個value。

一些其它的接口有Queue、Dequeue、SortedSet、SortedMap和ListIterator。

4.爲什麼Collection不從Cloneable和Serializable接口繼承?

Collection接口指定一組對象,對象即爲它的元素。如何維護這些元素由Collection的具體實現決定。例如,一些如List的Collection實現容許重複的元素,而其它的如Set就不容許。

不少Collection實現有一個公有的clone方法。然而,把它放到集合的全部實現中也是沒有意義的。這是由於Collection是一個抽象表現。重要的是實現。

當與具體實現打交道的時候,克隆或序列化的語義和含義才發揮做用。因此,具體實現應該決定如何對它進行克隆或序列化,或它是否能夠被克隆或序列化。點擊這裏一文學會序列化。

在全部的實現中受權克隆和序列化,最終致使更少的靈活性和更多的限制。特定的實現應該決定它是否能夠被克隆和序列化。點擊這裏一文學會序列化。

5.爲什麼Map接口不繼承Collection接口?

儘管Map接口和它的實現也是集合框架的一部分,但Map不是集合,集合也不是Map。所以,Map繼承Collection毫無心義,反之亦然。

若是Map繼承Collection接口,那麼元素去哪兒?Map包含key-value對,它提供抽取key或value列表集合的方法,可是它不適合「一組對象」規範。

6.Iterator是什麼?

Iterator接口提供遍歷任何Collection的接口。咱們能夠從一個Collection中使用迭代器方法來獲取迭代器實例。迭代器取代了Java集合框架中的Enumeration。迭代器容許調用者在迭代過程當中移除元素。

7.Enumeration和Iterator接口的區別?

Enumeration的速度是Iterator的兩倍,也使用更少的內存。Enumeration是很是基礎的,也知足了基礎的須要。可是,與Enumeration相比,Iterator更加安全,由於當一個集合正在被遍歷的時候,它會阻止其它線程去修改集合。

迭代器取代了Java集合框架中的Enumeration。迭代器容許調用者從集合中移除元素,而Enumeration不能作到。爲了使它的功能更加清晰,迭代器方法名已經通過改善。

8.爲什麼沒有像Iterator.add這樣的方法,向集合中添加元素?

語義不明,已知的是,Iterator的協議不能確保迭代的次序。然而要注意,ListIterator沒有提供一個add操做,它要確保迭代的順序。

9.爲什麼迭代器沒有一個方法能夠直接獲取下一個元素,而不須要移動遊標?

它能夠在當前Iterator的頂層實現,可是它用得不多,若是將它加到接口中,每一個繼承都要去實現它,這沒有意義。

10.Iterater和ListIterator之間有什麼區別?

(1)咱們可使用Iterator來遍歷Set和List集合,而ListIterator只能遍歷List。

(2)Iterator只能夠向前遍歷,而LIstIterator能夠雙向遍歷。

(3)ListIterator從Iterator接口繼承,而後添加了一些額外的功能,好比添加一個元素、替換一個元素、獲取前面或後面元素的索引位置。

11.經過迭代器fail-fast屬性,你明白了什麼?

每次咱們嘗試獲取下一個元素的時候,Iterator fail-fast屬性檢查當前集合結構裏的任何改動。若是發現任何改動,它拋出ConcurrentModificationException。Collection中全部Iterator的實現都是按fail-fast來設計的(ConcurrentHashMap和CopyOnWriteArrayList這類併發集合類除外)。

12.fail-fast與fail-safe有什麼區別?

Iterator的fail-fast屬性與當前的集合共同起做用,所以它不會受到集合中任何改動的影響。Java.util包中的全部集合類都被設計爲fail-fast的,

而java.util.concurrent中的集合類都爲fail-safe的。

Fall—fast迭代器拋出ConcurrentModificationException,

fall—safe迭代器從不拋出ConcurrentModificationException。

13.在迭代一個集合的時候,如何避免?

ConcurrentModificationException?

在遍歷一個集合的時候咱們可使用併發集合類來避免ConcurrentModificationException,好比使用CopyOnWriteArrayList,而不是ArrayList。

14.爲什麼Iterator接口沒有具體的實現?

Iterator接口定義了遍歷集合的方法,但它的實現則是集合實現類的責任。每一個可以返回用於遍歷的Iterator的集合類都有它本身的Iterator實現內部類。

這就容許集合類去選擇迭代器是fail-fast仍是fail-safe的。好比,ArrayList迭代器是fail-fast的,而CopyOnWriteArrayList迭代器是fail-safe的。

15.UnsupportedOperationException是什麼?

UnsupportedOperationException是用於代表操做不支持的異常。在JDK類中已被大量運用,在集合框架java.util.Collections.UnmodifiableCollection將會在全部add和remove操做中拋出這個異常。

16.hashCode和equals方法有何重要性?

HashMap使用Key對象的hashCode和equals方法去決定key-value對的索引。點擊這裏一文搞懂它們之間的關係。

當咱們試着從HashMap中獲取值的時候,這些方法也會被用到。若是這些方法沒有被正確地實現,在這種狀況下,兩個不一樣Key也許會產生相同的hashCode和equals輸出,HashMap將會認爲它們是相同的,而後覆蓋它們,而非把它們存儲到不一樣的地方。

一樣的,全部不容許存儲重複數據的集合類都使用hashCode和equals去查找重複,因此正確實現它們很是重要。equals和hashCode的實現應該遵循如下規則:

若是o1.equals(o2),那麼o1.hashCode == o2.hashCode老是爲true的。

若是o1.hashCode == o2.hashCode,並不意味着o1.equals(o2)會爲true。

17.Map接口提供了哪些不一樣的集合視圖?

Map接口提供三個集合視圖:

1)Set keyset:返回map中包含的全部key的一個Set視圖。集合是受map支持的,map的變化會在集合中反映出來,反之亦然。當一個迭代器正在遍歷一個集合時,若map被修改了(除迭代器自身的移除操做之外),迭代器的結果會變爲未定義。集合支持經過Iterator的Remove、Set.remove、removeAll、retainAll和clear操做進行元素移除,從map中移除對應的映射。

它不支持add和addAll操做。

2)Collection values:返回一個map中包含的全部value的一個Collection視圖。這個collection受map支持的,map的變化會在collection中反映出來,反之亦然。當一個迭代器正在遍歷一個collection時,若map被修改了(除迭代器自身的移除操做之外),迭代器的結果會變爲未定義。集合支持經過Iterator的Remove、Set.remove、removeAll、retainAll和clear操做進行元素移除,從map中移除對應的映射。它不支持add和addAll操做。

3)Set> entrySet:返回一個map鍾包含的全部映射的一個集合視圖。這個集合受map支持的,map的變化會在collection中反映出來,反之亦然。當一個迭代器正在遍歷一個集合時,若map被修改了(除迭代器自身的移除操做,以及對迭代器返回的entry進行setValue外),迭代器的結果會變爲未定義。集合支持經過Iterator的Remove、Set.remove、removeAll、retainAll和clear操做進行元素移除,從map中移除對應的映射。它不支持add和addAll操做。

18.HashMap和HashTable有何不一樣?

(1)HashMap容許key和value爲null,而HashTable不容許。

(2)HashTable是同步的,而HashMap不是。因此HashMap適合單線程環境,HashTable適合多線程環境。

(3)在Java1.4中引入了LinkedHashMap,HashMap的一個子類,假如你想要遍歷順序,你很容易從HashMap轉向LinkedHashMap,可是HashTable不是這樣的,它的順序是不可預知的。

(4)HashMap提供對key的Set進行遍歷,所以它是fail-fast的,但HashTable提供對key的Enumeration進行遍歷,它不支持fail-fast。

(5)HashTable被認爲是個遺留的類,若是你尋求在迭代的時候修改Map,你應該使用CocurrentHashMap。

19.如何決定選用HashMap仍是TreeMap?

對於在Map中插入、刪除和定位元素這類操做,HashMap是最好的選擇。然而,假如你須要對一個有序的key集合進行遍歷,TreeMap是更好的選擇。基於你的collection的大小,也許向HashMap中添加元素會更快,將map換爲TreeMap進行有序key的遍歷。

20.ArrayList和Vector有何異同點?

ArrayList和Vector在不少時候都很相似。

(1)二者都是基於索引的,內部由一個數組支持。

(2)二者維護插入的順序,咱們能夠根據插入順序來獲取元素。

(3)ArrayList和Vector的迭代器實現都是fail-fast的。

(4)ArrayList和Vector二者容許null值,也可使用索引值對元素進行隨機訪問。

如下是ArrayList和Vector的不一樣點。

(1)Vector是同步的,而ArrayList不是。然而,若是你尋求在迭代的時候對列表進行改變,你應該使用CopyOnWriteArrayList。

(2)ArrayList比Vector快,它由於有同步,不會過載。

(3)ArrayList更加通用,由於咱們可使用Collections工具類輕易地獲取同步列表和只讀列表。

21.Array和ArrayList有何區別?何時更適合用Array?

Array能夠容納基本類型和對象,而ArrayList只能容納對象。

Array是指定大小的,而ArrayList大小是固定的。

Array沒有提供ArrayList那麼多功能,好比addAll、removeAll和iterator等。儘管ArrayList明顯是更好的選擇,但也有些時候Array比較好用。

(1)若是列表的大小已經指定,大部分狀況下是存儲和遍歷它們。

(2)對於遍歷基本數據類型,儘管Collections使用自動裝箱來減輕編碼任務,在指定大小的基本類型的列表上工做也會變得很慢。

(3)若是你要使用多維數組,使用比List>更容易。

22.ArrayList和LinkedList有何區別?

ArrayList和LinkedList二者都實現了List接口,可是它們之間有些不一樣。

1)ArrayList是由Array所支持的基於一個索引的數據結構,因此它提供對元素的隨機訪問,複雜度爲O(1),但LinkedList存儲一系列的節點數據,每一個節點都與前一個和下一個節點相鏈接。因此,儘管有使用索引獲取元素的方法,內部實現是從起始點開始遍歷,遍歷到索引的節點而後返回元素,時間複雜度爲O(n),比ArrayList要慢。

2)與ArrayList相比,在LinkedList中插入、添加和刪除一個元素會更快,由於在一個元素被插入到中間的時候,不會涉及改變數組的大小,或更新索引。

3)LinkedList比ArrayList消耗更多的內存,由於LinkedList中的每一個節點存儲了先後節點的引用。

23.哪些集合類提供對元素的隨機訪問?

ArrayList、HashMap、TreeMap和HashTable類提供對元素的隨機訪問。

24.哪些集合類是線程安全的?

Vector、HashTable、Properties和Stack是同步類,因此它們是線程安全的,能夠在多線程環境下使用。Java1.5併發API包括一些集合類,容許迭代時修改,由於它們都工做在集合的克隆上,因此它們在多線程環境中是安全的。點擊這裏一文搞懂問什麼線程不安全。

25.併發集合類是什麼?

Java1.5併發包(java.util.concurrent)包含線程安全集合類,容許在迭代時修改集合。迭代器被設計爲fail-fast的,會拋出ConcurrentModificationException。一部分類爲:CopyOnWriteArrayList、 ConcurrentHashMap、CopyOnWriteArraySet。

26.隊列和棧是什麼,列出它們的區別?

棧和隊列二者都被用來預存儲數據。java.util.Queue是一個接口,它的實現類在Java併發包中。隊列容許先進先出(FIFO)檢索元素,但並不是老是這樣。Deque接口容許從兩端檢索元素。棧與隊列很類似,但它容許對元素進行後進先出(LIFO)進行檢索。Stack是一個擴展自Vector的類,而Queue是一個接口。

27.Collections類是什麼?

Java.util.Collections是一個工具類僅包含靜態方法,它們操做或返回集合。

它包含操做集合的多態算法,返回一個由指定集合支持的新集合和其它一些內容。這個類包含集合框架算法的方法,好比折半搜索、排序、混編和逆序等。

28.Comparable和Comparator接口有何區別?

Comparable和Comparator接口被用來對對象集合或者數組進行排序。Comparable接口被用來提供對象的天然排序,咱們可使用它來提供基於單個邏輯的排序。

Comparator接口被用來提供不一樣的排序算法,咱們能夠選擇須要使用的Comparator來對給定的對象集合進行排序。

29.咱們如何對一組對象進行排序?

若是咱們須要對一個對象數組進行排序,咱們可使用Arrays.sort方法。若是咱們須要排序一個對象列表,咱們可使用Collection.sort方法。

兩個類都有用於天然排序(使用Comparable)或基於標準的排序(使用Comparator)的重載方法sort。Collections內部使用數組排序方法,全部它們二者都有相同的性能,只是Collections須要花時間將列表轉換爲數組。

30.當一個集合被做爲參數傳遞給一個函數時,如何才能夠確保函數不能修改它?

在做爲參數傳遞以前,咱們可使用Collections.unmodifiableCollection(Collection c)方法建立一個只讀集合,

這將確保改變集合的任何操做都會拋出UnsupportedOperationException。

我本身收集了一些Java資料,裏面就包涵了一些BAT面試資料,以及一些 Java 高併發、分佈式、微服務、高性能、源碼分析、JVM等技術資料

須要獲取資料的朋友們能夠加Java羣:171662117,進羣幾個免費領取資料。
圖片描述
圖片描述

相關文章
相關標籤/搜索