在 Java 中的反射机制是指在运行状态中,对于任意一个类都能够知道这个类所有的属性和方法;并且对于任意一个对象,都能够调用它的任意一个方法;这种动态获取信息以及动态调用对象方法的功能成为 Java 语言的反射机制。
反射的应用场合
编译时类型和运行时类型
在 Java 程序中许多对象在运行是都会出现两种类型:编译时类型和运行时类型。 编译时的类型由声明对象时实用的类型来决定,运行时的类型由实际赋值给对象的类型决定 。如:
Person p=new Student();
其中编译时类型为 Person,运行时类型为 Student。
程序在运行时还可能接收到外部传入的对象,该对象的编译时类型为 Object,但是程序有需要调用该对象的运行时类型的方法。为了解决这些问题,程序需要在运行时发现对象和类的真实信息。然而,如果编译时根本无法预知该对象和类属于哪些类,程序只能依靠运行时信息来发现该对象和类的真实信息,此时就必须使用到反射了。
Java 反射 API
反射 API 用来生成 JVM 中的类、接口或则对象的信息。
- Class 类:反射的核心类,可以获取类的属性,方法等信息。
- Field 类:Java.lang.reflect 包中的类,表示类的成员变量,可以用来获取和设置类之中的属性值。
- Method 类: Java.lang.reflect 包中的类,表示类的方法,它可以用来获取类中的方法信息或
者执行方法。 - Constructor 类: Java.lang.reflect 包中的类,表示类的构造方法。
反射使用步骤(获取 Class 对象、调用对象方法)
- 获取想要操作的类的 Class 对象,他是反射的核心,通过 Class 对象我们可以任意调用类的方法。
- 调用 Class 类中的方法,既就是反射的使用阶段。
- 使用反射 API 来操作这些信息
获取 Class 对象的 3 种方法以及创建对象的两种方法、查看类的方法或属性等使用,详见代码注释:
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
/**
* @author ringleader
* @Description 验证反射知识点
* @date 2020/6/9
*/
public class TestReflect {
int size;
public int getSize() {
return size;
}
public void setSize(int size) {
this.size = size;
}
public TestReflect() {
}
public TestReflect(int size) {
this.size = size;
}
public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
TestReflect testReflect = new TestReflect();
/**
* 1、获取 Class 对象的 3 种方法
* */
// 调用某个对象的 getClass()方法
Class clazz = testReflect.getClass();
// 调用某个类的 class 属性来获取该类对应的 Class 对象
Class clazz1 = TestReflect.class;
// 使用 Class 类中的 forName()静态方法(最安全/性能最好)
Class clazz2 = Class.forName("知识点验证.反射.TestReflect");
/**
* 说明:检查异常 CheckedException:一般是外部错误,这种异常都发生在编译阶段,Java 编译器会强
* 制程序去捕获此类异常,要求你把这段可能出现异常的程序进行 try catch或throws异常类
* */
/**
* 2、查看类的方法或属性
* */
//获取TestReflect类的所有构造方法信息
Constructor[] declaredConstructors = clazz.getDeclaredConstructors();
for (Constructor declaredConstructor:declaredConstructors) {
System.out.println(declaredConstructor.toString());
}
//获取TestReflect类的所有方法信息
Method[] declaredMethods = clazz.getDeclaredMethods();
for (Method declaredMethod:declaredMethods){
System.out.println(declaredMethod.toString());
}
//获取TestReflect类的所有成员属性信息
Field[] fields = clazz.getDeclaredFields();
for (Field field:fields){
System.out.println(field.toString());
}
/**
* 3、创建对象的两种方法
* */
// 使用 Class 对象的 newInstance()方法来创建该 Class 对象对应类的实例,但是这种方法要求
//该 Class 对象对应的类有默认的空构造器。
TestReflect newReflectInstance = (TestReflect) clazz.newInstance();
//调用 Constructor 对象的 newInstance()
//先使用 Class 对象获取指定的 Constructor 对象,再调用 Constructor 对象的 newInstance()
//方法来创建 Class 对象对应类的实例,通过这种方法可以选定构造方法创建实例。
try {
Constructor declaredConstructor = clazz.getDeclaredConstructor(int.class);
TestReflect newReflectInstance2 = (TestReflect) declaredConstructor.newInstance(2);
System.out.println(newReflectInstance2.getSize());
} catch (NoSuchMethodException | InvocationTargetException e) {
e.printStackTrace();
}
}
}
参考:《JAVA核心知识点整理.pdf》