------Java培训、Android培训、iOS培训、.Net培训</a>、期待与您交流! -------
- Class类的使用
- 方法的反射
- 成员变量的反射
- 构造函数的反射
- 反射的应用:测试泛型的本质
通过Class,Method来认识泛型的本质
问题:我们学习了泛型都知道,泛型可以限制数据类型。学习过反射之后,是不是可以通过方法反射来操作,绕过编译,给指定泛型的方法添加另一种类型的数据?
先看代码
import java.util.*;
class MethodDemo
{
public static void main(String[]args)
{
ArrayList list=new ArrayList(); //先定义一个普通list集合
ArrayList<String>list1=new ArrayList<String>();//再定义一个加泛型的list集合
list1.add("hello");
//list.add(20); //错误
Class c1=list.getClass(); //分别获取两个集合的字节码。
Class c2=list2.getClass();
System.out.println(c1==c2); //比较字节码,打印结果为true
class MethodDemo
{
public static void main(String[]args)
{
ArrayList list=new ArrayList(); //先定义一个普通list集合
ArrayList<String>list1=new ArrayList<String>();//再定义一个加泛型的list集合
list1.add("hello");
//list.add(20); //错误
Class c1=list.getClass(); //分别获取两个集合的字节码。
Class c2=list2.getClass();
System.out.println(c1==c2); //比较字节码,打印结果为true
//------思考:既然加不加泛型的ArrayList对象字节码一致,那么我们能不能通过反射操作
//------给指定String类型的list1添加其他类型的数据?
/*
*c1==c2结果返回true说明编译之后集合的泛型是去泛型化的
*java中集合的泛型,是防止错误输入的,只在编译阶段有效
*绕过编译就无效了
*验证:可以通过方法反射来操作,绕过编译
*/
try
{
Method m=c2.getMethod("add",Objeci.class);
m.invoke(list1.size()); //绕过编译操作就绕过了泛型
System.out.println(list1.size()); //打印结果为2,说明加了一个元素进去
System.out.println(list1);
/*for(String str:list1)
{
System.out.println(str);
}*///现在不能这样遍历,因为list1集合中储存了int型数据。
}
catch(Exception e)
{
e.printStackTrace();
}
}
}
-------------------------------切割线要长-----------------------------
由此可知:
1、泛型的本质就是限制信息录入的类型,避免录入错误信息。
2、反射是可以绕过编译器的。
猜想:
虚拟机中并没有泛型类型对象,泛型类对象在编译器会全部变成普通类对象,所有的对象都是一样的,都属于普通的类。
(以上纯属个人见解,如果错误,敬请指点)