泛型擦除
Java 泛型的参数只可以代表类,不能代表个别对象。由于 Java 泛型的类型参数之实际类型在编译时会被消除,所以无法在运行时得知其类型参数的类型。Java 编译器在编译泛型时会自动加入类型转换的编码,故运行速度不会因为使用泛型而加快。-----百度百科
泛型擦除的体现
通过以下代码来感受以下什么是泛型擦除:
public class Erase {
public static void main(String[] args) {
// 创建两个不同泛型的list集合
List<Integer> integerList = new ArrayList<>();
List<String> stringList = new ArrayList<>();
System.out.println(integerList.getClass() == stringList.getClass());
System.out.println("integerList----> "+integerList.getClass().getName());
System.out.println("stringList----> "+stringList.getClass().getName());
}
}
输出结果
true
integerList----> java.util.ArrayList
stringList----> java.util.ArrayList
很明显发现,两个集合的泛型不同,但是比较class,返回的是true,并且通过反射获取到的className是java.util.ArrayList
不管是 ArrayList<Integer> 还是 ArrayList<String>,在编译完成后都会被编译器擦除成了 ArrayList。
Java 泛型擦除是 Java 泛型中的一个重要特性,其目的是避免过多的创建类而造成的运行时的过度消耗。所以,想 ArrayList<Integer> 和 ArrayList<String> 这两个实例,其实类实例是同一个。
可以把其他类型的数据放到指定了泛型的list集合中吗?
当然是可以的。因为泛型擦除的原因。完全可以将不相干的类型的数据存入指定泛型的集合中。
如下所示添加,肯定是要报错的,因为在编译的时候设定了规则,编译是不可能通过的。
按理说,因为泛型擦除的原因,完全是可以将不相干类型的数据放到指定泛型的集合中的。其实,可以换一种思路,当list集合运行起来之后,这个list集合因为泛型擦除是没有指定泛型的,即class是没有泛型的,所以可以在运行期间获取这个list集合,然后将值放到该list集合中。
public class Erase {
public static void main(String[] args) {
Erase erase = new Erase();
erase.testErase();
}
List<Integer> integerList = new ArrayList<>();
List<String> stringList = new ArrayList<>();
public void testErase(){
Double d = 10d;
try {
// 通过反射机制获取字段,然后将double类型的数据放入泛型为Integer集合中
Field integerField = this.getClass().getDeclaredField("integerList");
integerField.setAccessible(true);
List integerFieldList = (List) integerField.get(this);
integerFieldList.add(d);
// 通过反射机制获取字段,然后将double类型的数据放入泛型为String集合中
Field stringField = this.getClass().getDeclaredField("stringList");
stringField.setAccessible(true);
List stringFieldList = (List) stringField.get(this);
stringFieldList.add(d);
}catch (Exception e){
e.printStackTrace();
}
System.out.println("integerList ---> "+integerList);
System.out.println("stringList ---> "+stringList);
}
}
输出结果
integerList ---> [10.0]
stringList ---> [10.0]