一、反射概念
框架:半成品软件。可以在框架的基础上进行软件开发,简化编码。
反射:将java代码的各个组成部分封装为其他对象,可以在程序运行过程中操作这些对象,这就是java的反射机制。
反射的好处:
- 可以在程序运行过程中,操作这些对象。
- 可以解耦,提高程序的可扩展性。
二、获取Class对象的方式
获取class对象方式 | 作用 | 应用场景 |
---|---|---|
Class.forName(“全类名”) | 通过指定的字符串路径获取 | 多用于配置文件,将类名定义在配置文件中。 读取文件,加载类进内存,获取class对象 |
类名.class | 通过类名的属性class获取 | 多用于参数的传递 |
对象.getClass() | 通过对象的getClass()方法获取 getClass()方法在Object类中定义着 | 多用于对象的获取字节码的方式 |
注意:同一个字节码文件(*.class)在一次程序运行过程中,只会被加载一次,不论通过哪一种方式获取的
Class对象都是同一个
三、获取Class对象的信息
获取功能包括:
- 获取成员变量们
- 获取构造方法们
- 获取成员方法们
- 获取类名
下面分别介绍
1.获取成员变量们
Field类是成员变量类。
Field getField(String name) 返回一个 Field对象,该对象反映由该 Class对象表示的类或接口的指定公共成员字段。
Field[] getFields() 返回一个包含 Field对象的数组, Field对象反映由该 Class对象表示的类或接口的所有可访问的公共字段。
2.获取构造方法们
Constructor类是构造方法类,类中的每一个构造方法都是Constructor的对象,通过Constructor对象可以实例化对象。
Class类中与Constructor相关方法
- Constructor getConstructor(Class… parameterTypes)
根据参数类型获取构造方法对象,只能获得public修饰的构造方法。
如果不存在对应的构造方法,则会抛出 java.lang.NoSuchMethodException 异常。- Constructor getDeclaredConstructor(Class… parameterTypes)
根据参数类型获取构造方法对象,包括private修饰的构造方法。
如果不存在对应的构造方法,则会抛出 java.lang.NoSuchMethodException 异常。- Constructor[] getConstructors()
获取所有的public修饰的构造方法- Constructor[] getDeclaredConstructors()
获取所有构造方法,包括privat修饰的
Constructor类中常用方法
- T newInstance(Object… initargs) 根据指定参数创建对象。…代表可变参。
- void setAccessible(true) 暴力反射,设置为可以直接访问私有类型的构造方法
3.获取成员方法们
Method类是方法类,类中的每一个方法都是Method的对象,通过Method对象可以调用方法。
Class类中与Method相关方法
- Method getMethod(“方法名”, 方法的参数类型… 类型) 根据方法名和参数类型获得一个方法对象,只能是获取public修饰的
- Method getDeclaredMethod(“方法名”, 方法的参数类型… 类型) 根据方法名和参数类型获得一个方法对象,包括private修饰的
- Method[] getMethods() 获取所有的public修饰的成员方法,包括父类中。
- Method[] getDeclaredMethods() 获取当前类中所有的方法,包含私有的,不包括父类中。
Method类中常用方法
- Object invoke(Object obj, Object… args) 根据参数args调用对象obj的该成员方法,如果obj=null,则表示该方法是静态方法
- void setAccessible(boolean flag) 暴力反射,设置为可以直接调用私有修饰的成员方法
4.获取类名
String getSimpleName(); 获得简单类名,只是类名,没有包
String getName(); 获取完整类名,包含包名+类名
T newInstance() ; 创建此 Class 对象所表示的类的一个新实例。要求:类必须有public的无参数构造方法
三、反射应用举例
需求:写一个"框架",不能改变该类的任何代码的前提下,可以帮我们创建任意类的对象,并且执行其中任意方法
实现:
- 配置文件
- 反射
步骤:
- 将需要创建的对象的全类名和需要执行的方法定义在配置文件中
- 在程序中加载读取配置文件
- 使用反射技术来加载类文件进内存
- 创建对象
- 执行方法
pro.properties
className=com.testfan.domain.Student
methodName=sleep
Student.java
package com.testfan.domain;
public class Student {
public void sleep(){
System.out.println("sleep...");
}
}
RefectTest.java
public class ReflectTest {
@Test
public void test() throws Exception {
//可以创建任意类的对象,可以执行任意方法
//前提:不能改变该类的任何代码。可以创建任意类的对象,可以执行任意方法
//1.加载配置文件
//1.1创建Properties对象
Properties pro = new Properties();
//1.2加载配置文件,转换为一个集合
//1.2.1获取class目录下的配置文件
ClassLoader classLoader = ReflectTest.class.getClassLoader();
InputStream is = classLoader.getResourceAsStream("pro.properties");
pro.load(is);
//2.获取配置文件中定义的数据
String className = pro.getProperty("className");
String methodName = pro.getProperty("methodName");
//3.加载该类进内存
Class cls = Class.forName(className);
//4.创建对象
Object obj = cls.newInstance();
//5.获取方法对象
Method method = cls.getMethod(methodName);
//6.执行方法
method.invoke(obj);
}
}