目录:
1.Java中的泛型是什么 ? 使用泛型的好处是什么?
2.Java的泛型是如何工作的 ? 什么是类型擦除 ?
3.什么是泛型中的限定通配符和非限定通配符 ?
4.List extends T>和List super T>之间有什么区别 ?
5.如何编写一个泛型方法,让它能接受泛型参数并返回泛型类型?
6.Java中如何使用泛型编写带有参数的类?
7.编写一段泛型程序来实现LRU缓存?
8.你可以把List传递给一个接受List参数的方法吗?
9. Array中可以用泛型吗?
10. 如何阻止Java中的类型未检查的警告?
详解:
那些拥有Java1.4或更早版本的开发背景的人都知道,在集合中存储对象并在使用前进行类型转换是多么的不方便。泛型防止了那种情况的发生。它提供了编译期的类型安全,确保你只能把正确类型的对象放入集合中,避免了在运行时出现ClassCastException
。
2. Java的泛型是如何工作的 ? 什么是类型擦除 ?
泛型是通过类型擦除来实现的,编译器在编译时擦除了所有类型相关的信息,所以在运行时不存在任何类型相关的信息
。例如List在运行时仅用一个List来表示。这样做的目的
,是确保能和Java 5之前的版本开发二进制类库进行兼容。你无法在运行时访问到类型参数,因为编译器已经把泛型类型转换成了原始类型。
3. 什么是泛型中的限定通配符和非限定通配符 ?
限定通配符对类型进行了限制
。
有两种限定通配符:
泛型类型必须用限定内的类型来进行初始化,否则会导致编译错误
。
另一方面>表示了非限定通配符
,因为>可以用任意类型来替代。
4. List extends T>和List super T>之间有什么区别 ?
这两个List的声明都是限定通配符的例子,List extends T>可以接受任何继承自T的类型的List,而List super T>可以接受任何T的父类构成的List。例如List extends Number>可以接受List或List。
5. 如何编写一个泛型方法,让它能接受泛型参数并返回泛型类型?
编写泛型方法并不困难,你需要用泛型类型来替代原始类型,比如使用T, E or K,V等被广泛认可的类型占位符。最简单的情况下,一个泛型方法可能会像这样:
public V put(K key, V value) {
return cache.put(key, value);
}
面试官可能会要求你用泛型编写一个类型安全的类,而不是编写一个泛型方法。关键仍然是使用泛型类型来代替原始类型,而且要使用JDK中采用的标准占位符
。
7. 编写一段泛型程序来实现LRU缓存?
LinkedHashMap可以用来实现固定大小的LRU缓存,当LRU缓存已经满了的时候,它会把最老的键值对移出缓存
。LinkedHashMap提供了一个称为removeEldestEntry()的方法,该方法会被put()和putAll()调用来删除最老的键值对。
8. 你可以把List传递给一个接受List参数的方法吗?
不能!!!!
对任何一个不太熟悉泛型的人来说,这个Java泛型题目看起来令人疑惑,因为乍看起来String是一种Object,所以List应当可以用在需要List的地方,但是事实并非如此。真这样做的话会导致编译错误。如果你再深一步考虑,你会发现Java这样做是有意义的,因为List
。
List<Object> objectList;
List<String> stringList;
objectList = stringList; //compilation error incompatible types
public static void main(String[] args) {
List<String> list = new ArrayList<String>();
list.add("测试1");
list.add("测试2");
List<Object> result = new ArrayList<Object>();
for(int i =0;i<list.size();i++){
result.add(list.get(i));//可以赋值
}
test1(list);//但是作为参数传递的话 会报错//在idea中这行已经报红了,报错了;
test2(list);//如果这样用 就可以了
System.out.println(result.toString());
}
public static String test1(List<Object> t){
return "";
}
public static String test2(List<? extends Object> t){
return "";
}
不能!!!
Array事实上并不支持泛型,这也是为什么Joshua Bloch在Effective Java一书中建议使用List来代替Array,因为List可以提供编译期的类型安全保证,而Array却不能
。
虚拟机本身的实现就不支持泛型数组
。因为数组是协变,类型擦除后就没法满足数组协变的原则
。
协变是指比如说一个int类型的数组可以插入一个double类型的数值,编译能够通过,但是运行时会出错。因为数组内建了编译期和运行时的检查;但是在使用泛型时,类型信息在编译期被擦除了,运行时也就无从检查
。因此,泛型将这种错误检测移入到编译期
10. 如何阻止Java中的类型未检查的警告?
如果你把泛型和原始类型混合起来使用,例如下列代码,Java 5的javac编译器会产生类型未检查的警告,例如
List<String> rawList = new ArrayList()
特别感谢并参考:
https://github.com/CyC2018/CS-Notes/blob/master/notes/Java%20%E5%9F%BA%E7%A1%80.md