1、反射机制概述以及应用场景
首先,何为反射机制:
JAVA反射机制是在运行状态中,对于任意一个类 (class文件),都能够知道这个类的所有属性和方法;
对于任意一个对象,都能够调用它的任意一个方法和属性;
这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
动态获取类中信息,就是java反射,可以理解为对类的解剖。注意,这里操作的不是“.java”文件,而是字节码文件“.class”。那么反射机制有什么用处?(35-28-01-2.45解析)
要想要对字节码文件进行解剖,必须要有字节码文件对象。 如何获取字节码文件对象呢?(见35-28-01-7.50解析)
这种反射机制大大提高了程序的扩展性,如下Tomcat服务器的例子
注意,配置文件不是往里面写类,而是往里面写我们想要运行的类的信息。(35-day28-02-2.50)配置文件里面配置的其实就是应用程序里面变化的参数信息,而反射就是我们要运行的类是变化的(不确定,因为不同的对象的类有其自己具体的实现),那么我们将这个类写入配置文件。
那么我们如何使用java的反射机制来获取“.class”文件对象呢?(35-day28-02-7.30)java使用Class类来对它的字节码文件“.class”进行描述,Class类提供了一些方法来获取字节码文件当中的信息。
2、反射的使用——获取字节码文件对象的3种方法
先看看下面代码
package reflect.demo;
import test.demo.Person;//注意得将这个包的的Person类导入才可以使用
public class ReflectDemo {
/**
* @param args
* @throws ClassNotFoundException
*/
public static void main(String[] args) throws ClassNotFoundException {
getClassObject_3();
}
/*
* 获取字节码对象的方式:
* 1、Object类中的getClass()方法。
* 想要用这种方式,必须要明确具体的类,并创建对象,较为麻烦
*/
public static void getClassObject_1()
{
//任何对象的创建都要依赖字节码,因此我们可以通过对象来获取其字节码文件对象。
Person p = new Person();
Class clazz = p.getClass();
Person p1 = new Person();
Class clazz1 = p1.getClass();
System.out.println(clazz == clazz1);
/*
结果:
person run
person run
true
创建Person类无参对象,Person的无参构造方法会被调用。
另一方面,不同Person对象所获取的Class对象是相同的,因为2个Person对象的创建都是依赖同一个“.class”文件
*/
}
/*
* 方式二:
* 2、任何数据类型都具备一个静态的属性——.class,利用这个静态属性来获取其对应的Class对象。
* 相对简单,但是还是要明确用到类中的静态成员,还是不够扩展。
*/
public static void getClassObject_2()
{
Class clazz = Person.class;
Class clazz1 = Person.class;
System.out.println(clazz == clazz1);
/*
结果是:true
*/
}
/*
* 方式三:
* 只要通过给定的类的字符串名称就可以获取该类,更为扩展。可用Class类中的forName方法完成。
* 这种方式只要有名称即可,更为方便,扩展性更强。
*/
public static void getClassObject_3() throws ClassNotFoundException
{
//这种方式,我们就可以将要实现的类的名称写入配置文件中,而不需要知道这个类是怎么实现的
String className = "test.demo.Person";//先获取类名称的字符串
//static Class<?> forName(String className) 返回与带有给定字符串名的类或接口相关联的 Class 对象。
//系统会通过forName()方法自动在该项目下的“.classpath”文件下寻找,而.classpatn会让其去bin文件查找字符串所指向的类,
//因此这个类必须在当前的项目下,在其他项目下会报出异常
//另外,类名必须带上包名,这样系统才能查找到相应的类,否则查找不到。
//虽然导包,但是导包导入Person类,而我们这里使用的是字符串。我们发现不用导入Person类同样可以执行该程序。
Class clazz = Class.forName(className);
System.out.println(clazz);
/*
结果是:class test.demo.Person
*/
}
}
3、获取Class中的构造函数
我们获取到Person类的.class文件,这与直接new一个Person对象有什么区别?见下面的代码:
package reflect.demo;
import java.lang.reflect.Constructor;
public class ReflectDemo {
/**
* @param args
* @throws Exception
*/
public static void main(String[] args) throws Exception {
createNewObject_2();
}
//创建空参数的对象
public static void createNewObject() throws ClassNotFoundException, InstantiationException, IllegalAccessException
{
// 早期:new时候,先根据被new的类的名称找寻该类的字节码文件,并加载进内存,
// 并创建该字节码文件对象,并接着创建该字节文件的对应的Person对象
test.demo.Person p = new test.demo.Person();
//现在:
String name = "test.demo.Person";
//根据类的名称,找寻该名称.class文件,并加载进内存,并产生Class对象。
Class