泛型常见⾯试题
1. 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. 如何编写⼀个泛型⽅法,让它能接受泛型参数并返回泛型类型?
编写泛型⽅法并不困难,你需要⽤泛型类型来替代原始类型,⽐如使⽤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) (0); //需要显式的类型转换
item = (String) (1); //抛ClassCastException,因为Integer不能被转换为String
List<String> listOfString = new ArrayList();
listOfString.add(“abcd”);
listOfString.add(1234); //编译错误,⽐在运⾏时抛异常要好
item = (0); //不需要显式的类型转换 – 编译器⾃动转换
这些都是Java泛型⾯试中频繁出现的问题及其答案。所有这些⾯试题都不困难,其实它们都是基于泛型的基础知识。任何对泛型有不错了解的Java程序员都肯定熟知这些泛型题⽬。如果你有任何好的⾯试题,不管是在什么⾯试中碰到的,或者如果你想知道任何Java泛型⾯试题的答案。