该文首发于我的个人博客ReadLamp
前言
反射是什么?
Java反射机制是在运行状态中对于任意一个类,都能知道这个类的所以属性和方法;对于任何一个对象,都能够调用它的任何一个方法和属性;这样动态获取新的以及动态调用对象方法的功能就叫做反射。
有什么用?
可以将类的各个部分封装为其他对象,在程序运行过程当中操作这些对象。反射是实现框架的基石。
反射可以解耦,提高程序的可拓展性。
本文主要分为以下部分:
- 获取Class对象
- Class对象的方法
- 实例演示
获取Class对象
要想使用反射,首先需要获取Class对象。常用的获取方法有3种,但无论是哪一种方式获取的Class对象都是同一个,因为同一个字节码文件(*.class)在一次程序运行过程中,只会被加载一次。这个Class对象是由JVM生成的,通过它能够获悉整个类的结构。
常用的获取方法如下:
- 使用Class类的静态方法。
Class.forName("全类名/包名.类名")
:将字节码文件加载进内存,返回Class对象。
例如Class.forName(“java.lang.String”);
- 使用类的.class语法。
类名.class
:通过类名的属性class获取。
例如String.class;
- 使用对象的getClass()方法。
对象.getClass()
:getClass()方法在Object类中定义着。
例如students.getClass();
Class对象的方法
主要介绍Class对象的获取功能。一般常用的获取有四类:成员变量,方法,构造方法,类名。
获取成员变量(数组)
对应的方法
- Field[] getFields() :获取所有public修饰的成员变量
- Field getField(String name) 获取指定名称的 public修饰的成员变量
- Field[] getDeclaredFields() 获取所有的成员变量,不考虑修饰符
- Field getDeclaredField(String name)
获取后的操作方法
- 获取属性名
- String getName()
- 设置值
- void set(Object obj, Object value)
- 获取值
- get(Object obj)
- 忽略访问权限修饰符的安全检查
- setAccessible(true):暴力反射
获取方法(数组)
对应的方法
- Method[] getMethods()
- Method getMethod(String name, 类<?>… parameterTypes)
- Method[] getDeclaredMethods()
- Method getDeclaredMethod(String name, 类<?>… parameterTypes)
获取后的操作方法
- 执行方法
- Object invoke(Object obj, Object… args):需要对象参数
- 获取方法名称
- String getName()
获取构造方法(数组)
对应的方法
- Constructor<?>[] getConstructors()
- Constructor getConstructor(类<?>… parameterTypes)
- Constructor getDeclaredConstructor(类<?>… parameterTypes)
- Constructor<?>[] getDeclaredConstructors()
获取后的操作方法
- 创建对象
- T newInstance(Object… initargs)
- 空参简化
- Class对象的newInstance方法
获取类名
对应的方法
- String getName()
实例演示
需求
写一个“框架”,在不能改变该类的任何代码的前提下,可以帮我们创建任意类的对象,并且执行其中任意方法。
实现
- 使用配置文件
- 反射
步骤
- 将需要创建的对象的全类名和需要执行的方法定义在配置文件中
- 在程序中加载读取配置文件
- 使用反射技术来加载类文件进内存
- 创建对象
- 执行方法
相关知识点
- Properties类方法
- load(Stream):加载流
- getProperty(String 配置属性):获取配置属性的字符串
- 获取构造器
- 类.class.getClassLoader()
- 获取读取流
- 构造器对象.getResourceAsStream(“配置文件名”);
实例代码
配置文件代码:
//配置文件名为pro.properties
className = com.readlamp.demo.Calculator
methodName = start
框架类代码:
public class ReflectTest {
public static void main(String[] args) throws Exception {
// 加载配置文件
// 获取配置文件对象
Properties properties = new Properties();
// 获得构造器
ClassLoader classLoader = ReflectTest.class.getClassLoader();
// 读取流
InputStream resourceAsStream = classLoader.getResourceAsStream("pro.properties");
// 加载至配置文件对象中
properties.load(resourceAsStream);
// 获取数据
String className = properties.getProperty("className");
String methodName = properties.getProperty("methodName");
// 加载至内存
Class cla = Class.forName(className);
// 创建对象
Object o = cla.newInstance();
// 获取方法对象
Method method = cla.getMethod(methodName);
// 执行对象方法
method.invoke(o);
}
}
参考资料:
- AndyBao:https://blog.csdn.net/buddyuu/article/details/52458241