反射作为JAVA特有的一个机制,有必要了解以及掌握。那么首先我们得清楚反射是什么,什么时候使用反射,反射能做什么。
提到反射,就不得不联系到Java程序的运行状态,它分为两种:
1.编译时:通过Javac命令,生成一个或多个.class字节码文件(每个.class文件对应一个类)
2.运行时:通过Java命令,将生成的一个或多个.class字节码文件加载到内存中。(由JVM提供的类加载器完成)
什么时候使用反射?
我们会将描述现实生活中的一类事物,用类进行描述,若需要具体到一个,那么通过new关键字创建一个实例即可,再通过调用该实例的方法以及操作属性完成我们想要的功能。这种情况下,我们实际上是先知道我们想要创建什么类的实例,所以直接new某个类的实例,调用该实例的方法,比如:
Person person=new Person();
person.setAge();
我们明确的想创建一个person类,以及调用其中的setAge()方法,所以直接新建对象再调用方法即可,
但在某些情况下,我们需要得知并且使用一个编译完全未知的类,并且创建一个对象,调用方法等等。
比如我们在对数据库进行操作时,比如查询,假设我们需要查询顾客信息、人员信息、员工信息,那么按照通常的情况,我们需要分别写下三个带这三种返回值的方法,有几个查询我们就需要写几个方法。这样显然是很麻烦的。如果我们能写一个通用的查询方法,在方法中对我们需要查询的信息进指定,那么就方便很多了。这时,我们就需要用到反射。
既然叫反射,那么它的过程必定有一个类似反射的过程,这里我们可以用图来表示,首先我们看创建一个对象的过程:
当我们创建一个对象(比如Person p=new Person();)时,首先会在方法区里加载类的信息,比如该类的属性、方法、实现的接口等等,接下来p会存在栈中,通过new关键字在堆中创建一个person的实例。
而反射的过程:
首先我们知道,类是将生活中一类有共性的事物进行描述的,比如Person人类,张三李四王五都是人,他们都有名字性别等共性,所以都用Person类进行描述,再比如Animal动物类,猫狗都是动物,也有相同的共性,而Person与Animal两个类,抛开其属性不说,两者也有共性,比如都有构造器、都有父类等等,他们也有一个用于描述自己的类:Class类。现在我们将图变成这样:
在上面我们说到,方法区在新建类时会加载类的所有信息,而Class类是拥有所有类的共性,也就说明两者是等同的,在反射中,在加载类信息的同时,JVM会为我们在堆中自动生成一个Class实例,而该实例因为包含了该类的所有信息,包括构造器,既然可以获取到构造器,那么就可以生成一个我们想要的Person实例,这就是一个反射,在图中我们也可以清晰地看到这一个反射的流程,那么由此我们可以看出,得到一个Class的实例,是反射开启的关键。
创建Class的实例有如下几种种:
1.通过运行时类的属性class,比如Person类:
//1.通过运行时类的属性class
Class clazz1=Person.class;
System.out.println(clazz1);
结果:
2.通过运行时类的getClass()方法:
//2.通过运行时类对象的getClass()方法
Person p=new Person();
Class clazz2=p.getClass();
System.out.println(clazz2);
3.通过Class类的静态方法forName(String className),这里需要填入全类名:
//3.通过Class类的静态方法forName(String className)
String className="com.lya.Reflection.Person";
try {
Class clazz3=Class.forName(className);
System.out.println(clazz3);
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
由于可能会出现找不到类异常,需要进行异常处理
4.通过类加载器:
//4.通过类加载器
ClassLoader c1=this.getClass().getClassLoader();
try {
Class clazz4=c1.loadClass(className);
System.out.println(clazz4);
} catch (ClassNotFoundException e) {
e.printStackTrace();
}