反射的概念
反射就是在程序运行期间能够动态获取类的全部属性和方法并且能操作对象的全部属性和方法的一种Java语言机制。
反射的本质
要知道反射的本质,我们先要了解类的加载机制,如下图所示。
类的加载机制:当我们new一个Student对象时,jvm会从磁盘中寻找该类的.class文件并加载到jvm内存中,jvm会自动创建一个Class对象,并且每个类都只会创建一个class对象。
反射的本质就是获得类的Class对象,再反向去获取类或者对象的各种属性和方法。
获得Class对象的三种方式
在使用反射时,我们需要先获取Class对象。获取Class对象有三种方式:
第一,Object对象.getClass()
第二,任何数据类型(包括基本数据类型)都有一个静态static的class属性
第三,通过Class类的静态方法forName(String className)来获取 Class.forName(String className) ------>[这是我们常用的获取类对象的方法]
需要注意的是,在程序运行期间,一个类只有一个Class对象产生,该Class对象由JVM管理。获得了类对象,我们就随之可以使用这个类对象的所有属性和方法。
Student student=new Student();
Class stuClass=student.getClass();
Class stuClass2=Student.class;
System.out.println(stuClass==stuClass2);
Class stuClass3=Class.forName("com.teasir.reflect.Student");
System.out.println(stuClass3==stuClass2);
输出:
true
true
反射生效期间
通过反射,我们可以获取一个类的任意方法(包括构造方法)和属性,从而充分利用一个类所提供的一切资源。另外,我们在使用各类集合(如List,Set,Map)时,为了指明集合中元素的类型,通常会用到泛型,泛型是作用于程序编译期,编译过后,泛型的约束就会失效;反射则作用于程序编译期之后的运行期,因此,我们可以通过反射来越过泛型的检查。我们通过下面的案例来看:
/**
* 泛型作用于编译期,会在编译之前对程序进行检查
* 编译期过后,泛型约束失效
* 反射作用于运行期
* 演示通过反射越过泛型的检查
*/
public static class Test {
public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
//1.先声明一个List集合,元素泛型约束为String类型
List<String> strList=new ArrayList<>();
//2.向集合中添加String型的元素
strList.add("aa");
strList.add("bb");
//注:在编译期之前,向集合中添加非泛型元素,程序会在编译期就报错
//strList.add(123);
//这时,我们通过反射来越过泛型检查
Class strListClass = strList.getClass();//先获取Class对象
//获取add方法,为了保障程序清晰,在main方法上抛出异常
Method method = strListClass.getMethod("add", Object.class);
//通过反射调用add方法,添加一个int型数据
method.invoke(strList,123);
System.out.println("strList="+strList);
}
}