java反射:框架设计的灵魂
java代码的三个阶段:
框架:半成品软件。可以在框架的基础上进行软件开发,简化编码
反射:将类的各个组成部分封装为其他对象,这就是反射机制
好处:
1. 可以在程序运行过程中,操作这些对象。
2. 可以解耦,提高程序的可扩展性。
获取Class对象的方式:
1. Class.forName("全类名"):将字节码文件加载进内存,返回Class对象
* 多用于配置文件,将类名定义在配置文件中。读取文件,加载类
2. 类名.class:通过类名的属性class获取
* 多用于参数的传递
3. 对象.getClass():getClass()方法在Object类中定义着。
* 多用于对象的获取字节码的方式
结论:
同一个字节码文件(*.class)在一次程序运行过程中,只会被加载一次,
不论通过哪一种方式获取的Class对象都是同一个。
public void ClassObjectTest() throws ClassNotFoundException { //1. Class.forName("全类名"):将字节码文件加载进内存,返回Class对象 // 多用于配置文件,将类名定义在配置文件中。读取文件,加载类 Class cls1=Class.forName("com.clss.test.Person"); System.out.println(cls1); /* * 2. 类名.class:通过类名的属性class获取 * 多用于参数的传递 * */ Class cls2= Person.class; System.out.println(cls2); /* * 3. 对象.getClass():getClass()方法在Object类中定义着。 * 多用于对象的获取字节码的方式。 * */ Person p=new Person(); Class cls3=p.getClass(); System.out.println(cls3); //测试是否是同一对象 System.out.println(cls1==cls2); System.out.println(cls1==cls3); System.out.println(cls2==cls3); /* *结论: *同一个字节码文件(*.class)在一次程序运行过程中, * 只会被加载一次,不论通过哪一种方式获取的Class对象都是同一个。 */ }
`==使用clss对象功能:
=获取功能:
1. 获取成员变量们:
* Field[] getFields() :获取所有public修饰的成员变量
* Field getField(String name) 获取指定名称的 public修饰的成员变量
* Field[] getDeclaredFields() 获取所有的成员变量,不考虑修饰符
* Field getDeclaredField(String name)
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()
Field:成员变量
* 操作:
1. 设置值
* void set(Object obj, Object value)
2. 获取值
* get(Object obj)
3. 忽略访问权限修饰符的安全检查
* setAccessible(true):暴力反射
测试代码:
public void ReflectTest() throws NoSuchFieldException, IllegalAccessException { /*获取Person类的class的对象*/ Class personClass =Person.class; /* * 1. 获取成员变量们: * Field[] getFields() :获取所有public修饰的成员变量 * Field getField(String name) 获取指定名称的 public修饰的成员变量 * Field[] getDeclaredFields() 获取所有的成员变量,不考虑修饰符 * Field getDeclaredField(String name) * */ Field[] fields=personClass.getFields(); for(Field field:fields){ System.out.println(field); } System.out.println("==========================="); System.out.println(personClass.getField("address")); System.out.println("================================"); Field[] fields2=personClass.getDeclaredFields(); for(Field field:fields2){ System.out.println(field); } System.out.println("==========================="); Field d=personClass.getDeclaredField("name"); d.setAccessible(true);/*暴力反射*/ /*即使直接获得私有的值也不会报错*/ Person p=new Person("test",20); Object value=d.get(p); System.out.println(value); }
public void reflectConstrTest() throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException { Class classPerson=Person.class; /* * 2. 获取构造方法们 * Constructor<?>[] getConstructors() * Constructor<T> getConstructor(类<?>... parameterTypes) * Constructor<T> getDeclaredConstructor(类<?>... parameterTypes) * Constructor<?>[] getDeclaredConstructors() * */ Constructor[] constructor=classPerson.getConstructors(); for(Constructor str:constructor){ System.out.println(str); } System.out.println("========================================="); /*获得对应的构造函数*/ Constructor constructor1=classPerson.getDeclaredConstructor(String.class,int.class); System.out.println(constructor1); //创建对象 Object person=constructor1.newInstance("张三",123); System.out.println(person); }
反射的案例:
* 需求:写一个"框架",不能改变该类的任何代码的前提下,可以帮我们创建任意类的对象,并且执行其中任意方法
* 实现:
1. 配置文件
2. 反射
* 步骤:
1. 将需要创建的对象的全类名和需要执行的方法定义在配置文件中
2. 在程序中加载读取配置文件
3. 使用反射技术来加载类文件进内存
4. 创建对象
5. 执行方法
Student类
package com.clss.test; public class Student { public void eat(){ System.out.println("eat............"); } }
pro.properties
className=com.clss.test.Student
methodName=eat
实现方式
package com.test.demo1; import java.io.IOException; import java.io.InputStream; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.util.Properties; public class ReflectTest { public static void main(String[] args) throws IOException, ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException { //1.加载配置文件 //1.1创建Properties对象 Properties pro=new Properties(); //1.2加载配置文件,转换为一个集合 //1.2.1获取class目录下的配置文件 /*使用类加载器完成*/ ClassLoader classLoader=MainTest.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); } }