基本概念
反射机制允许程序在执行期间借助于ReflectionAPI取得任何类的内部信息(成员变量、构造器、成员方法等),并操作对象的属性和方法。
加载完类之后,在堆中就产生了一个Class类型的对象,包含了类的完整结构信息
Java反射机制可以完成
- 在运行时判断任意一个对象所属的类
- 在运行时构造任意一个类的对象
- 在运行时得到任意一个类所具有的成员变量和方法
- 在运行时调用任意一个对象的成员变量和方法
- 生成动态代理
反射机制的主要类
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-FY3k9lTE-1647694879032)(D:\Typora\Java学习\image-20220318183709637.png)]
代码实现
public static void main(String[] args) throws Exception {
//创建Properties对象
Properties properties = new Properties();
//输入流读取配置文件
properties.load(new BufferedReader(new FileReader("D:\\JavaProject\\JavaSE\\practice01\\src\\re.properties")));
String classPath = properties.getProperty("name");
String methodName = properties.getProperty("method");
//使用反射机制加载类
Class cls = Class.forName(classPath);
//通过加载的类获取对象实例
Object o = cls.newInstance();
System.out.println("运行类型为" + o.getClass());
Method method = cls.getMethod(methodName);
System.out.println("===============================");
//调用对象的方法:方法.invoke(对象)
method.invoke(o);
//getField不能得到私有的字段
Field nameField = cls.getField("age");
//获取属性值:成员变量对象.get(对象)
System.out.println(nameField.get(o));
//反射机制获取类的无参构造器
Constructor constructor = cls.getConstructor();
System.out.println(constructor);
//反射机制获取类的有参构造器,传输参数为构造器指定参数的类对象
Constructor constructor1 = cls.getConstructor(String.class);
System.out.println(constructor1);
}
反射的优点和缺点
优点:可以动态地创建和使用对象,框架的底层核心,使用灵活。
缺点:使用反射基本是解释执行,对执行速度有影响,比正常创建类访问 要慢
method.setAccessible(true);
//在反射调用方法时,取消检查访问,可提高一定执行效率
Class类
- class类也是一个类,也继承自Object类
- Class类对象不是new出来的,而是通过ClassLoder类的loadClass( )创建
- 对于某个类的Class类对象,在内存中只有一份,因为类只加载一次
- 每个对象实例都会记得自己是由哪个Class实例生成的
- 通过Class对象可以完整得到一个类的完整结构
- Class对象是存放在堆中的
- 类加载时同时为产生类的字节码二进制数据,存放在方法区,有的地方称为类的元数据
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-IH9JHYjf-1647694879034)(D:\Typora\Java学习\image-20220318214801472.png)]
Class类常用方法
String classAllPath = "reflection.Car";
Class<?> cls = Class.forName(classAllPath);
//显示是哪个类的Class对象 class reflection.Car
System.out.println(cls);
//输出Class对象的运行类型 class java.lang.Class
System.out.println(cls.getClass());
//输出该类所在的包名 reflection
System.out.println(cls.getPackage().getName());
//输出全类名 reflection.Car
System.out.println(cls.getName());
//通过Class对象创建Car对象实例
Car car = (Car) cls.newInstance();
//通过toString方法,输出car的属性
System.out.println(car);
//通过反射获取Car中的属性对象brand public java.lang.String reflection.Car.brand
Field brand = cls.getField("brand");
System.out.println(brand);
//通过反射获取属性的值 奔驰
System.out.println(brand.get(car));
//通过反射给对象属性赋值 宝马
brand.set(car, "宝马");
System.out.println(brand.get(car));
//获取所有属性字段的名称 brand price color
Field[] fields = cls.getFields();
for (Field field : fields) {
System.out.print(field.getName());
}
}
获取Class类对象的六种方式
已知类的全类名,且该类在类路径下,可通过Class类的静态方法**forName(全类名)**获取,多用于配置文件,读取类全路径,加载类
Class cls1 = Class.forName("reflection.Car");
已知具体的类,通过类的class获取,多用于参数传递,比如反射得到对应的构造器
Class<Car> cls2 = Car.class;
已知某个类的实例,调用该实例的getClass方法获取Class对象,多用于通过创建好的对象,获取Class对象
Car car = new Car();
Class<Car> cls3 = car.getClass();
基本数据类型对应的包装类可通过 .TYPE得到Class对象
Class<Integer> IntegerClass = Integer.TYPE;
哪些类型有Class对象
- 外部类,成员内部类,静态内部类,局部内部类,匿名内部类
- interface:接口
- 数组
- enum:枚举
- annotation:注解
- 基本数据类型
- void
类加载
反射机制是Java实现动态语言的关键,也就是通过反射实现类动态加载
静态加载:编译时加载相关的类,如果找不到该类则报错,依赖性太强
动态加载:运行时加载需要的类,如果运行时不使用该类,即使不存在该类也不会报错,依赖性降低
类加载时机
创建对象时(new)静态加载
子类被加载时,父类也加载 静态加载
调用类中的静态成员时 静态加载
通过反射机制 动态加载