1. 程式人生 > >Java面試題-泛型篇

Java面試題-泛型篇

發生 是你 hashmap 參數化 實例變量 圖片 使用 多態 來替

139. Java中的泛型是什麽 ? 使用泛型的好處是什麽?

泛型是Java SE 1.5的新特性,泛型的本質是參數化類型,也就是說所操作的數據類型被指定為一個參數。

好處:

1、類型安全,提供編譯期間的類型檢測

2、前後兼容

3、泛化代碼,代碼可以更多的重復利用

4、性能較高,用GJ(泛型JAVA)編寫的代碼可以為java編譯器和虛擬機帶來更多的類型信息,這些信息對java程序做進一步優化提供條件。

140,Java的泛型是如何工作的 ? 什麽是類型擦除 ?如何工作?

1、類型檢查:在生成字節碼之前提供類型檢查

2、類型擦除:所有類型參數都用他們的限定類型替換,包括類、變量和方法(類型擦除)

3、如果類型擦除和多態性發生了沖突時,則在子類中生成橋方法解決

4、如果調用泛型方法的返回類型被擦除,則在調用該方法時插入強制類型轉換

類型擦除:

所有類型參數都用他們的限定類型替換:

比如T->Object ? extends BaseClass->BaseClass

如何工作:

泛型是通過類型擦除來實現的,編譯器在編譯時擦除了所有類型相關的信息,所以在運行時不存在任何類型相關的信息。例如 List<String>在運行時僅用一個List來表示。這樣做的目的,是確保能和Java 5之前的版本開發二進制類庫進行兼容。你無法在運行時訪問到類型參數,因為編譯器已經把泛型類型轉換成了原始類型。根據你對這個泛型問題的回答情況,你會得到一些後續提問,比如為什麽泛型是由類型擦除來實現的或者給你展示一些會導致編譯器出錯的錯誤泛型代碼。

141,你可以把List<String>傳遞給一個接受List<Object>參數的方法嗎?

對任何一個不太熟悉泛型的人來說,這個Java泛型題目看起來令人疑惑,因為乍看起來String是一種Object,所以 List<String>應當可以用在需要List<Object>的地方,但是事實並非如此。真這樣做的話會導致編譯錯誤。如果你再深一步考慮,你會發現Java這樣做是有意義的,因為List<Object>可以存儲任何類型的對象包括String, Integer等等,而List<String>卻只能用來存儲String s。

List<Object> objectList;

List<String> stringList;

objectList = stringList; //compilation error incompatible types

142,如何阻止Java中的類型未檢查的警告?

如果你把泛型和原始類型混合起來使用,例如下列代碼,java 5的javac編譯器會產生類型未檢查的警告,例如

List<String> rawList = newArrayList()

註意: Hello.java使用了未檢查或稱為不安全的操作;

這種警告可以使用@SuppressWarnings(“unchecked”)註解來屏蔽。

143,Java中List<Object>和原始類型List之間的區別?

原始類型和帶參數類型<Object>之間的主要區別是,在編譯時編譯器不會對原始類型進行類型安全檢查,卻會對帶參數的類型進行檢查,通過使用Object作為類型,可以告知編譯器該方法可以接受任何類型的對象,比如String或Integer。

這道題的考察點在於對泛型中原始類型的正確理解。它們之間的第二點區別是,你可以把任何帶參數的類型傳遞給原始類型List,但卻不能把List<String>傳遞給接受 List<Object>的方法,因為會產生編譯錯誤。

144,編寫一段泛型程序來實現LRU緩存?

對於喜歡Java編程的人來說這相當於是一次練習。給你個提示,LinkedHashMap可以用來實現固定大小的LRU緩存,當LRU緩存已經滿了的時候,它會把最老的鍵值對移出緩存。

LinkedHashMap提供了一個稱為removeEldestEntry()的方法,該方法會被put() 和putAll()調用來刪除最老的鍵值對。當然,如果你已經編寫了一個可運行的JUnit測試,你也可以隨意編寫你自己的實現代碼。

145,Array中可以用泛型嗎?

這可能是Java泛型面試題中最簡單的一個了,當然前提是你要知道Array事實上並不支持泛型,這也是為什麽Joshua Bloch在Effective Java一書中建議使用List來代替Array,因為List可以提供編譯期的類型安全保證,而Array卻不能。

146,如何編寫一個泛型方法,讓它能接受泛型參數並返回泛型類型?

編寫泛型方法並不困難,你需要用泛型類型來替代原始類型,比如使用T, E or K,V等被廣泛認可的類型占位符。最簡單的情況下,一個泛型方法可能會像這樣:

public V put(K key, V value) {

return cahe.put(key,value);

}

147,C++模板和java泛型之間有何不同?

java泛型實現根植於“類型消除”這一概念。當源代碼被轉換為Java虛擬機字節碼時,這種技術會消除參數化類型。有了Java泛型,我們可以做的事情也並沒有真正改變多少;他只是讓代碼變得漂亮些。鑒於此,Java泛型有時也被稱為“語法糖”。

這和 C++模板截然不同。在 C++中,模板本質上就是一套宏指令集,只是換了個名頭,編譯器會針對每種類型創建一份模板代碼的副本。

由於架構設計上的差異,Java泛型和C++模板有很多不同點:

C++模板可以使用int等基本數據類型。Java則不行,必須轉而使用Integer。

在Java中,可以將模板的參數類型限定為某種特定類型。

在C++中,類型參數可以實例化,但java不支持。

在Java中,類型參數不能用於靜態方法(?)和變量,因為它們會被不同類型參數指定的實例共享。在C++,這些類時不同的,因此類型參數可以用於靜態方法和靜態變量。

在Java中,不管類型參數是什麽,所有的實例變量都是同一類型。類型參數會在運行時被抹去。在C++中,類型參數不同,實例變量也不同。

Java知音公眾號推送Java開發中的一些必備技能,包括但不限於數據庫、Java核心、流行框架、管理工具以及服務器相關,同時為您精選流行的開源項目,必會的面試選題,練手項目,優質視頻資源等。讓您閑暇之余鞏固一下自己的知識,不知不覺中提高自己的開發水平。

技術分享圖片

Java面試題-泛型篇