android面试 系统复习系列(一)泛型
泛型的本质
泛型的本质是为了参数化类型。也就是说在泛型使用过程中,操作的数据类型被指定为一个参数,这种参数类型可以用在类、接口和方法中,分别被称为泛型类、泛型接口、泛型方法。
也就是把类本身的类型当参数传进去,而不是在定义的时候就写死。
泛型的特型
泛型擦除
Java 的泛型在编译器有效,在运行期被删除,就是说所有泛型参数类型在编译后都会被清除掉
举两个例子
List<String> stringList = new ArrayList<String>();
List<Integer> integerList = new ArrayList<Integer>();
Class classStringList = stringList.getClass();
Class classIntegerList = integerList.getClass();
if(classStringList.equals(classIntegeList)){
Log.d("泛型测试","类型相同");
}
输出结果:D/泛型测试: 类型相同
public class Test {
public void dealList(List<String> stringList){
}
public void dealList(List<Integer> intList) {
}
}
看起来没什么问题,但是编译器却报错,大概就是方法签名重复。
为什么 JVM 要做泛型擦除
我觉得是为了节省 JVM 运行期间的逻辑,提高运行效率。
泛型实例不兼容
举个例子,看一段代码
void fun(List<Object> list) {
}
void main() {
List<String> list = new List<String>();
fun(list)
}
乍一看这段代码似乎没什么问题,但是编译器却会报错。
其实仔细想想是可以理解的,因为List<Objec>可以存储任何类型的对象,所以如果 fun 方法是下面这样实现的话,那 main 方法中的传参肯定是不行的
void fun(List<Object> list) {
list.add(1);
}
那假设我的方法需要List<String> 和 List<Integer> 两种类型参数怎么办,这时候就引出了另一个概念,通配符
void fun(List<?> list) {
}
这样就可以都传了。
精心挑选的面试题
1、什么泛型擦除,为什么要这样做
2、List<? extends E> 在编译后会变成什么样
List<? extends E> 擦除后的类型为 List<E>
3、你可以把List< String>传递给一个接受List< Object>参数的方法吗,为什么
4、List<? extends T>和List <? super T>之间有什么区别
就是限定传入类型的上限和下限