Java的反射技术

什么是反射机制

Java的反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能调用它的任意属性和方法。这种动态获取信息以及动态调用对象属性和方法的即使称为Java的反射机制。

反射的应用场景

在编译时根本无法知道该类或对象可能属于哪些类,程序只能依靠运行时信息来发现该类和对象的真实信息。、

反射的作用

通过反射可以使程序代码访问装载到JVM中的类的内部信息。
获取已装载类的属性信息
获取已装载类的方法
获取已装载类的构造方法信息
在JDK中,反射机制主要通过以下类来实现的,这些类除了Class类以外都位于java.lang.reflect包中
java.lang.Class <T> 类
java.lang.reflect.Field 类
java.lang.reflect.method 类
java.lang.reflect.Constructor <T> 类

java.lang.Class <T> 类

1)Class类是Java反射机制的起源和入口

Class类继承自Object类

用于获取与类相关的各种信息

提供了获取类信息的相关方法

2)Class类是所有类的共同的图纸

每个类都有自己的对象,好比图纸和实物的关系

每个类也可以看做是一个对象,有共同的图纸Class,存放类的结构信息,能够通过相应方法取出相应信息

类的名字

属性

方法

构造方法

父类和接口

3)Class类的常用方法

getFields()  // 获得类的public类型的属性
getDeclaredFields()  // 获得类的所有属性
getField(String name)  // 获得类的指定属性
getMethods()  // 获得类的public类型的方法
getMethos(String name, Class<?>...parameterTypes)  // 获得类的指定方法
getConstructors()  // 获得类的public类型的构造方法
getConstructor(Class<?>...parameterTypes)  // 获得类的指定的构造方法
getName()  // 获得类的完成的名字
getpackage()  // 获得此类所属的包
getSuperclass()  // 获得此类的父类对应的Class实例对象
isArray() // 判断该类型是否是数组
isEnum() // 判断该类型是否是接口
isInterface() // 判断该类型是否是接口
isPrimitive() // 判断该类型是否是基本数据类型(8种基本数据类型) newInstance()
// 通过类的无惨构造方法创建一个对象

4)获取类的Class实例对象的方法(可分为三类)

Class.forName(“类的全限定名”)  // 调用Class类的静态方法,该方法将装载类,并做类的静态初始化,返回一个Class类的实例对象
类名.class  // JVM将使用类装载器,将类装入内存(前提是类还没有转入内存),不做类的初始化工作,返回类的Class的对象实例。
包装类.TYPE  // 包装类的Class类型的静态常量,可以得到相应基本数据类型的Class实例对象
对象名.getClass()  // Object类的方法,对垒进行静态初始化,非静态初始化,返回代表该对象的运行时类的Class实例对象
对象名.getSuperclass  //  Class类的方法,返回该对象的父类的Class类实例对象

说明:类的静态属性和方法的初始化是在加载类的时候完成了;而类的非静态属性和方法的初始化时在new一个实例对象的时候完成的。

当我们编译一个新的Java类时,JVM就会帮我们编译成Class对象实例,存放在类名.class文件中。在运行时,当需要生成这个类的对象时,JVM就会检查这个类是否已经装载到内存中,若没有装载,则把类名.class文件装载到内存,若已经加载了,则根据类名.class文件生成Class实例对象。

在Java中,任何一个类都会有一个Class实例对象与之对应,在这个Class对象中,保存着实例化该类时所需的基本信息。如A.class返回的就是类A的Class对象。

1 public class Test {
2     public static void main(String [] args) throws ClassNotFoundException {
3         Class clazz = Class.forName("Test");  // Class.forName()
4         System.out.println(clazz.getName());
5         System.out.println(Test.class.getName());  // 类名.class
6         Test test = new Test();
7         System.out.println(test.getClass().getName());  // 对象名.getClass()
8     }
9 }

使用反射技术创建对象(Constructor类)

1)通过Class类的newInstance()方法

该方法要求该Class实例对象的对应类有无参构造方法。执行newInstance()实际就是实行类的无参构造方法来创建该类的实例对象。

2)通过Constructor类的newInstance()方法

首先使用Class对象获取指定的Constructor对象;然后调用Constructor对象的newInstance()方法创建Class对象对应类的实例对象。

使用该方法可以选择使用指定的构造方法来创建实例对象。

使用反射技术查询修改属性值(Field类)

通过Class对象的getFields()方法返回Filed类型的数组,从而获得该类所包含的全部的Field属性

通过Class对象的getField(String name)方法返回Field类型的实例对象,从而获得该类指定的Field属性

每个Field实例对象对应一个属性,可以通过Field类提供的以下方法来访问或修改相应属性

getXxx(Object obj)  // 获取obj对象该Field的属性值。此处的Xxx对应8种基本数据类型,如果该属性类型是引用类型则直接使用get(Object obj)

setXxx(Object obj, Xxx val)  // 将obj对象的该Field赋值为val。此处的Xxx对应8种基本数据类型,如果该属性类型是引用类型则直接使用set(Object obj, Object val)

setAccessible(Boolean flag)  // 若flag为true,则取消属性的访问权限控制,即使private属性也可以进行访问

使用反射技术动态调用方法(Method类)

通过Class对象的getMethods()方法返回Method类型的数组,从而获得该类所包含的全部的方法

通过Class对象的getMethod(String name, Class<?>...parameterTypes)方法返回Method类型的实例对象,从而获得该类指定的方法

每个Method对象对应一个方法,获得Method对象后,可以通过调用Method类的invoke()方法来调用对应的方法

public Object invoke(Object obj, Object...args) throws IllegalAccessException, IllegalArgumentException, InvocationTargetException  // obj代表当前方法所属的对象的名字, args代表当前方法的参数列表,返回值Object是当前方法的返回值,即执行当前方法的结果。

原文地址:https://www.cnblogs.com/0820LL/p/9697490.html