反射:基础概念
框架:半成品软件。
反射:将类的各个组成部分封装为其他对象,这就是反射机制。
光看概念可能不太好理解,我们先来看看java代码的三个阶段:
- 源代码阶段
- Class类对象阶段
- 运行时阶段
如上图,两个红色框中的内容。将javac编译获得的字节码文件加载进内存,并将类中定义的 成员变量,构造函数,成员方法 分别封装为 Field Constructor Method对象。这就是放射机制。
好处:
- 可以在程序运行过程中,操作这些对象。
- 可以解耦,提供程序的可扩展性。
获取Class对象的方法
1: 第一阶段,Class.forName("全限定类名"),将字节码文件加载进内存,返回CLass对象。
当我们写完Person。class文件保存后,IDEA就自动编译了文件,字节码文件就存在了。
多用于配置文件,将类名定义在配置文件中,读取文件,加载类
2: 第二阶段,通过类名的属性class获取,类名.class.
多用于参数的传递
3: 第三阶段,对象.getClass()方法,该方法在Object类中定义着。
多用于使用对象获取字节码的方式
结论:同一个字节码文件(*.class)再一次程序运行过程中,只会被加载一次,不论通过哪一种方式获取的Class对象都是同一个。
Class对象功能
1:获取成员变量们
Field[] getFields(); //获取所有public修饰的成员变量
Field getField(String name); //获取指定名称的 public修饰的成员变量
Field[] getDeclaredFields(); //获取所有的成员变量,不考虑修饰符
Field getDeclaredField(String name); //获取指定名称的成员变量,不考虑修饰符
Field:成员变量
设置值
void set(Obejct obj, Object value);
获取值
Obejct get(Obejct obj);
忽略访问权限修饰符的安全检查
setAccessible(true);//暴力反射
2:获取构造方法们
Constructor<?>[] getConstructors();
Constructor<T> getConstructor(Class<?>... parameterTypes);
Constructor<?>[] getDeclaredConstructors();
Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes);
Constructor:构造方法
创建对象
T newInstance(Object... initargs);
如果使用空参数的构造方法创建对象,操作可以简化为:Class对象的newInstance方法
3:获取成员方法们
Method[] getMethods();
Method getMethod(String name, Class<?>... parameterTypes);
Method[] getDeclaredMethods();
Method getDeclaredMethod(String name, Class<?>... parameterTypes);
Method:方法对象
执行方法
Object invoke(Object obj, Object... args);
获取方法名称
String getName();
4:获取类名
String getName();
案例(反射的魅力)
需求:实现一个“框架”类,在不改变任何代码的前提下,可以创建任意对象,并执行任意方法。
实现:
- 配置文件
- 反射
步骤:
1:将需要创建的对象的全限定类名和需要执行的方法名称定义在配置文件中。
2:在程序中加载读取配置文件
3:使用反射技术加载类文件进内存
4:创建对象
5:执行方法
代码: