一、什么是反射
在运行状态中,对于 任意一个类,都能够获取到这个类的所有属性和方法;对于 任意一个对象,都能够调用它的任意一个方法和属性,这种动态获取信息以及调用对象方法的功能就称为Java
的反射机制。
要使用反射机制,就必须先获取到该类的 字节码文件对象 ,通过字节码文件对象,能够通过该类中的方法获取到我们需要的所有信息。
当类加载的时候,会将.class
文件读入内存,并为它创建一个Class
对象,它包含了所有相关的信息,也就是上面所说的 字节码文件对象。
二、获取 Class 对象的方法
- 通过
Class
类的静态方法,此时该类还是处于 源文件阶段。
Class clazz1 = Class.forName("限定名");
- 当类被加载成
.class
文件时,此时Clazz
类变成了.class
,在获取该字节码文件对象,也就是获取自己,该类处于 字节码阶段。
Class clazz2 = Clazz.class;
- 通过类的实例获取该类的字节码文件对象,该类处于 创建对象阶段。
Class clazz3 = p.getClass();
三、使用反射
在我们使用反射的时候会涉及到三个方面:构造函数、属性和方法,以下面这个类为例。
/**
* @author lizejun
**/
public class ReflectHolder {
public String openValue;
private String closeValue;
private ReflectHolder() {}
public ReflectHolder(String openValue, String closeValue) {
this.openValue = openValue;
this.closeValue = closeValue;
}
public void setOpenValue(String openValue) {
this.openValue = openValue;
}
private void setCloseValue(String closeValue) {
this.closeValue = closeValue;
}
}
3.1 构造函数
通过Class
类获取构造方法,有以下四类:
-
Constructor[] - getConstructors()
:所有public
构造方法。 -
Constructor[] - getDeclaredConstructors()
:所有构造方法,包括public
、protected
、private
、默认。 -
Constructor - getConstructor(null)
:无参构造方法。 -
Constructor - getDeclaredConstructor(x.class)
:参数为x
的构造方法。
通过其中一种方法,我们可以获取到Constrcutor
对象,并通过该对象来创建实例:
-
newInstance(Object... args)
:使用此Constructor
对象表示的构造方法来创建其对应类的实例,并用指定的参数args
初始化该实例。
3.2 属性
-
Field[] - getFields()
:所有public
属性。 -
Field[] - getDeclaredFields()
:所有属性。 -
Field - getField(属性名称)
:获取对应名称的Field
。
获取到Field
对象后,可以通过Field.set(obj, value)
方法,设置obj
对象的对应属性值为value
。
对于非public
属性,需要先调用Field. setAccessible(true)
解除私有规定。
3.3 方法
-
Method[] - getMethods()
:所有public
方法。 -
Method[] - getDeclaredMethods()
:所有方法。 -
Method - getMethod("方法名", a.class, b.class, ...)
:通过方法名和形参列表.class
这两项来唯一确定方法。
获取到Method
对象后,可以通过Method.invoke(obj, 实参列表)
来调用该方法,其返回值就为该方法的返回值,需要进行类型强制转换。
对于不可访问的方法,需要先调用Method.setAccessible(true)
解除私有规定。