泛型常见面试题

1. Java中的泛型是什么 ? 使用泛型的好处是什么?java

这是在各类Java泛型面试中,一开场你就会被问到的问题中的一个,主要集中在初级和中级面试中。那些拥有Java1.4或更早版本的开发背景的人 都知道,在集合中存储对象并在使用前进行类型转换是多么的不方便。泛型防止了那种状况的发生。它提供了编译期的类型安全,确保你只能把正确类型的对象放入 集合中,避免了在运行时出现ClassCastException。程序员

2. Java的泛型是如何工做的 ? 什么是类型擦除 ?面试

这是一道更好的泛型面试题。泛型是经过类型擦除来实现的,编译器在编译时擦除了全部类型相关的信息,因此在运行时不存在任何类型相关的信息。例如 List<String>在运行时仅用一个List来表示。这样作的目的,是确保能和Java 5以前的版本开发二进制类库进行兼容。你没法在运行时访问到类型参数,由于编译器已经把泛型类型转换成了原始类型。根据你对这个泛型问题的回答状况,你会 获得一些后续提问,好比为何泛型是由类型擦除来实现的或者给你展现一些会致使编译器出错的错误泛型代码。请阅读个人Java中泛型是如何工做的来了解更 多信息。编程

3. 什么是泛型中的限定通配符和非限定通配符 ?缓存

这是另外一个很是流行的Java泛型面试题。限定通配符对类型进行了限制。有两种限定通配符,一种是<? extends T>它经过确保类型必须是T的子类来设定类型的上界,另外一种是<? super T>它经过确保类型必须是T的父类来设定类型的下界。泛型类型必须用限定内的类型来进行初始化,不然会致使编译错误。另外一方面<?>表 示了非限定通配符,由于<?>能够用任意类型来替代。更多信息请参阅个人文章泛型中限定通配符和非限定通配符之间的区别。安全

4. List<? extends T>和List <? super T>之间有什么区别 ?框架

这和上一个面试题有联系,有时面试官会用这个问题来评估你对泛型的理解,而不是直接问你什么是限定通配符和非限定通配符。这两个List的声明都是 限定通配符的例子,List<? extends T>能够接受任何继承自T的类型的List,而List<? super T>能够接受任何T的父类构成的List。例如List<? extends Number>能够接受List<Integer>或List<Float>。在本段出现的链接中能够找到更多信息。测试

5. 如何编写一个泛型方法,让它能接受泛型参数并返回泛型类型?spa

编写泛型方法并不困难,你须要用泛型类型来替代原始类型,好比使用T, E or K,V等被普遍承认的类型占位符。泛型方法的例子请参阅Java集合类框架。最简单的状况下,一个泛型方法可能会像这样:对象

public V put(K key, V value) {

return cache.put(key, value);

}

6. Java中如何使用泛型编写带有参数的类?

这是上一道面试题的延伸。面试官可能会要求你用泛型编写一个类型安全的类,而不是编写一个泛型方法。关键仍然是使用泛型类型来代替原始类型,并且要使用JDK中采用的标准占位符。

7. 编写一段泛型程序来实现LRU缓存?

对于喜欢Java编程的人来讲这至关因而一次练习。给你个提示,LinkedHashMap能够用来实现固定大小的LRU缓存,当LRU缓存已经满 了的时候,它会把最老的键值对移出缓存。LinkedHashMap提供了一个称为removeEldestEntry()的方法,该方法会被put() 和putAll()调用来删除最老的键值对。固然,若是你已经编写了一个可运行的JUnit测试,你也能够随意编写你本身的实现代码。

8. 你能够把List<String>传递给一个接受List<Object>参数的方法吗?

对任何一个不太熟悉泛型的人来讲,这个Java泛型题目看起来使人疑惑,由于乍看起来String是一种Object,因此 List<String>应当能够用在须要List<Object>的地方,可是事实并不是如此。真这样作的话会致使编译错误。如 果你再深一步考虑,你会发现Java这样作是有意义的,由于List<Object>能够存储任何类型的对象包括String, Integer等等,而List<String>却只能用来存储Strings。

List<Object> objectList;

List<String> stringList;

objectList = stringList; //compilation error incompatible types

9. Array中能够用泛型吗?

这多是Java泛型面试题中最简单的一个了,固然前提是你要知道Array事实上并不支持泛型,这也是为何Joshua Bloch在Effective Java一书中建议使用List来代替Array,由于List能够提供编译期的类型安全保证,而Array却不能。

10. 如何阻止Java中的类型未检查的警告?

若是你把泛型和原始类型混合起来使用,例以下列代码,Java 5的javac编译器会产生类型未检查的警告,例如

List<String> rawList = new ArrayList()

注意: Hello.java使用了未检查或称为不安全的操做;

这种警告可使用@SuppressWarnings(“unchecked”)注解来屏蔽。

Java泛型面试题补充更新:

我手头又拿到了几个Java泛型面试题跟你们分享下,这几道题集中在泛型类型和原始类型的区别上,以及咱们是否能够用Object来代替限定通配符的使用等等:

Java中List<Object>和原始类型List之间的区别?

原始类型和带参数类型<Object>之间的主要区别是,在编译时编译器不会对原始类型进行类型安全检查,却会对带参数的类型进行检 查,经过使用Object做为类型,能够告知编译器该方法能够接受任何类型的对象,好比String或Integer。这道题的考察点在于对泛型中原始类 型的正确理解。它们之间的第二点区别是,你能够把任何带参数的类型传递给原始类型List,但却不能把List<String>传递给接受 List<Object>的方法,由于会产生编译错误。更多详细信息请参阅Java中的泛型是如何工做的。

Java中List<?>和List<Object>之间的区别是什么?

这道题跟上一道题看起来很像,实质上却彻底不一样。List<?> 是一个未知类型的List,而List<Object> 实际上是任意类型的List。你能够把List<String>, List<Integer>赋值给List<?>,却不能把List<String>赋值给 List<Object>。     

List<?> listOfAnyType;

List<Object> listOfObject = new ArrayList<Object>();

List<String> listOfString = new ArrayList<String>();

List<Integer> listOfInteger = new ArrayList<Integer>();

listOfAnyType = listOfString; //legal

listOfAnyType = listOfInteger; //legal

listOfObjectType = (List<Object>) listOfString; //compiler error – in-convertible types

想了解更多关于通配符的信息请查看Java中的泛型通配符示例

List<String>和原始类型List之间的区别.

该题相似于“原始类型和带参数类型之间有什么区别”。带参数类型是类型安全的,并且其类型安全是由编译器保证的,但原始类型List却不是类型安全 的。你不能把String以外的任何其它类型的Object存入String类型的List中,而你能够把任何类型的对象存入原始List中。使用泛型的 带参数类型你不须要进行类型转换,可是对于原始类型,你则须要进行显式的类型转换。

List listOfRawTypes = new ArrayList();

listOfRawTypes.add(“abc”);

listOfRawTypes.add(123); //编译器容许这样 – 运行时却会出现异常

String item = (String) listOfRawTypes.get(0); //须要显式的类型转换

item = (String) listOfRawTypes.get(1); //抛ClassCastException,由于Integer不能被转换为String

List<String> listOfString = new ArrayList();

listOfString.add(“abcd”);

listOfString.add(1234); //编译错误,比在运行时抛异常要好

item = listOfString.get(0); //不须要显式的类型转换 – 编译器自动转换

这些都是Java泛型面试中 频繁出现的问题及其答案。全部这些面试题都不困难,其实它们都是基于泛型的基础知识。任何对泛型有不错了解的Java程序员都确定熟知这些泛型题目。若是 你有任何好的面试题,不论是在什么面试中碰到的,或者若是你想知道任何Java泛型面试题的答案。

相关文章
相关标签/搜索