反射:框架设计的灵魂
框架:半成品软件。可以在框架的基础上进行软件开发,简化编码。
概述
反射:将类的各个组成部分封装为其他对象,这就是反射机制。
好处:
1. 可以在程序运行过程中,操作这些对象。
2. 可以解耦,提高程序的可扩展性。
编译完成后,.java和.class 都属于源代码阶段, 程序运行,.class被类加载器加载到内存中时为Class类对象阶段,new出对象后为Runtime阶段。
获取Class对象的方式
- Class.forName("全类名"):将字节码文件加载进内存,返回Class对象,多用于配置文件,将类名定义在配置文件中。读取文件加载类。
- 类名.class:通过类名的属性class获取,多用于参数的传递
- 对象.getClass():getClass()方法在Object类中定义着,多用于对象的获取字节码的方式
同一个字节码文件(*.class)在一次程序运行过程中,只会被加载一次,不论通过哪一种方式获取的Class对象都是同一个。
Class对象功能
获取功能:
1. 获取成员变量们
- Field[] getFields() :获取所有public修饰的成员变量
- Field getField(String name) 获取指定名称的 public修饰的成员变量
- Field[] getDeclaredFields() 获取所有的成员变量,不考虑修饰符
- Field getDeclaredField(String name) 获取指定名称的 public修饰的成员变量,不考虑修饰符
2. 获取构造方法们
- Constructor<?>[] getConstructors()
- Constructor<T> getConstructor(类<?>... parameterTypes)
- Constructor<T> getDeclaredConstructor(类<?>... parameterTypes)
- Constructor<?>[] getDeclaredConstructors()
3. 获取成员方法们:
- Method[] getMethods()
- Method getMethod(String name, 类<?>... parameterTypes)
- Method[] getDeclaredMethods()
- Method getDeclaredMethod(String name, 类<?>... parameterTypes)
4. 获取全类名
- String getName() Class对象功能:
Field:成员变量
操作:
- void set(Object obj, Object value) 设置值
- get(Object obj) 获取值
- setAccessible(true):暴力反射,忽略访问权限修饰符的安全检查。通常在使用private修饰的变量和方法前,要通过方法对象的该方法,setAccessible(true) ,将其权限打开。
Constructor:构造方法
创建对象:如果使用空参数构造方法创建对象,操作可以简化:Class对象的newInstance方法
- T newInstance(Object... initargs)
Method:方法对象
执行方法:
- Object invoke(Object obj, Object... args)
获取方法名称:
- String getName:获取方法名
demo
需求:写一个"框架",不能改变该类的任何代码的前提下,可以帮我们创建任意类的对象,并且执行其中任意方法
实现:
1. 配置文件
2. 反射
步骤:
1. 将需要创建的对象的全类名和需要执行的方法定义在配置文件中
2. 在程序中加载读取配置文件
3. 使用反射技术来加载类文件进内存
4. 创建对象
5. 执行方法
public class Person {
private String name;
private int age;
public String a;
protected String b;
String c;
private String d;
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
", a='" + a + '\'' +
", b='" + b + '\'' +
", c='" + c + '\'' +
", d='" + d + '\'' +
'}';
}
public void eat(){
System.out.println("eat...");
}
public void eat(String food){
System.out.println("eat..."+food);
}
}
配置文件
className=cn.hello.Student
methodName=eat
反射
public static void main(String[] args) 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);
}
执行结果
eat...
感谢关注
加油啊!