共计 4163 个字符,预计需要花费 11 分钟才能阅读完成。
1、反射概念
反射就是在程序运行的过程中,动态的创建对象。
只要知道该类的类名称,就可以使用它的字节码对象创建该类的一个对象。对于这个类中的任何一个成员或者属性,我们都可以访问或者使用它。
2、Class 对象
2.1、什么是字节码文件?
字节码文件,也就是以.class 结尾的文件,他是 Java 中的类经过编译之后生成的字节码数据文件!该文件最终会被 JVM 加载进内存。
2.2、什么是字节码对象?
前面说过,JVM 最终会把字节码文件加载进内存,该文件进 JVM 内存之后,Java 就会把他当做是一个字节码对象!因为 Java 是一门面向对象的语言,在 Java 的世界中,所有的东西都是对象,自然也包括该字节码文件了!
2.3、字节码对象的获取方式
(1)Object 对象的 getClass 方法
(2)类型.class 属性
(3)Class 类的静态方法–>forName()
public static void main(String[] args) throws Exception {// 方式一 | |
//Object getClass()方法 | |
Person p = new Person(); | |
Class clazz1 = p.getClass(); | |
// 方式二 | |
// 类型.class 属性 | |
Class clazz2 = Person.class; | |
Class clazz3 = Student.class; | |
Class clazz4 = int.class;// 基本数据类型也有字节码对象 | |
// 方式三 | |
//Class 类的静态方法 ---> forName("类的完整路径"); | |
// 注意:这里要用全路径名 | |
Class clazz5 = Class.forName("Test.Student"); | |
// 思考:同一个类的字节码对象,在程序的一次运行过程中会被获取几次? | |
} |
2.4、Class 对象的组成
Class 对象主要由以下几部分组成:
(1)若干 Constructor 对象
(2)若干 Method 对象
(3)若干 Field 对象
3、Constructor 对象
3.1、Constructor 概述
首先思考一个问题,构造器的作用是什么?构造器的作用就是构造一个对象!我们平时用 new 关键来创建一个类的对象,new 后面跟着的其实就该类的构造方法!如下图所示:
构造器是用来构建对象的,同样的,构造器对象 /Constructor 对象也是用来构建对象的!
public class Person {private String name; | |
private int age; | |
public Person() { } | |
public Person(String name, int age) {this.name = name; | |
this.age = age; | |
} | |
public String toString() {return "Person{" + | |
"name='" + name + '\'' + | |
", age=" + age + | |
'}'; | |
} | |
} |
3.2、得到构造器对象
(1)公共 构造器的获取方法
getConstructors()
public static void main(String[] args) throws Exception {// 获得字节码对象 | |
Class clazz = Person.class; | |
// 得到所有公共构造器对象 | |
Constructor[] constructors = clazz.getConstructors(); | |
for (Constructor constructor : constructors) {System.out.println(constructor); | |
} | |
} | |
(2)所有构造器的获取方式
getDeclaredConstructors()
public static void main(String[] args) throws Exception {// 获得字节码对象 | |
Class clazz = Person.class; | |
// 得到所有公共构造器对象 | |
Constructor[] constructors = clazz.getDeclaredConstructors(); | |
for (Constructor constructor : constructors) {System.out.println(constructor); | |
} | |
} |
(3)获得指定公共构造器
getConstructor(Class… parameter);
public static void main(String[] args) throws Exception {// 获得字节码对象 | |
Class clazz = Person.class; | |
// 得到所有公共构造器对象 | |
Constructor constructor = clazz.getConstructor(String.class,int.class); | |
// 参数是可变形参,参数类型是该参数类型的字节码对象 --?Class… parameter | |
System.out.println(constructor); | |
} |
public static void main(String[] args) throws Exception {// 获得字节码对象 | |
Class clazz = Person.class; | |
// 得到所有公共构造器对象 | |
Constructor constructor = clazz.getConstructor(); | |
// 参数是可变形参,参数类型是该参数类型的字节码对象 --?Class… parameter | |
System.out.println(constructor); | |
} |
(4)获得指定非公共构造器对象
getDeclaredConstructor(Class… parameter);
public static void main(String[] args) throws Exception {// 获得字节码对象 | |
Class clazz = Person.class; | |
// 得到所有公共构造器对象 | |
Constructor constructor = clazz.getDeclaredConstructor(String.class,int.class); | |
// 参数是可变形参,参数类型是该参数类型的字节码对象 --?Class… parameter | |
System.out.println(constructor); | |
} |
public static void main(String[] args) throws Exception {// 获得字节码对象 | |
Class clazz = Person.class; | |
// 得到所有公共构造器对象 | |
Constructor constructor = clazz.getDeclaredConstructor(); | |
// 参数是可变形参,参数类型是该参数类型的字节码对象 --Class… parameter | |
System.out.println(constructor); | |
} |
3.3、用构造器对象创建该类实例
newInstance(Object… initargs));
4、Method 对象
4.1、Method 对象的获取
getMethods()—> 得到所有公共的方法对象
getDeclaredMethods()-? 得到所有的方法对象
getMethod(String name,Class… parameter); 得到指定的公共方法对象
getDeclaredMethod(String name,Class… parameter); 得到指定的任意方法对象
public static void main(String[] args) throws Exception {// 获得字节码对象 | |
Class clazz = Person.class; | |
// 得到所有公共的方法对象 | |
Method[] methods = clazz.getMethods(); | |
// 得到所有方法对象 | |
Method[] methods2 = clazz.getDeclaredMethods(); | |
// 得到指定的公共方法对象 | |
Method method = clazz.getMethod("method", String.class); | |
// 得到指定的非公共方法对象 | |
Method method2 = clazz.getDeclaredMethod("method", String.class); | |
} |
4.2、Method 对象的使用
invoke(Object obj, Object… args);
public static void main(String[] args) throws Exception {// 获得字节码对象 | |
Class clazz = Person.class; | |
// 得到指定的公共方法对象 | |
Method method = clazz.getMethod("method", String.class); | |
// 正常调用方法的格式是:对象名. 方法名(参数) | |
// 反射执行方法的格式:方法对象.invoke(方法所属对象, 实参列表); | |
method.invoke(clazz.newInstance(), "haha"); | |
} |
5、Field 对象
5.1、获取 Field 对象
public static void main(String[] args) throws Exception {// 获得字节码对象 | |
Class clazz = Person.class; | |
// 得到指定的 Field | |
Field field = clazz.getField("name"); | |
System.out.println(field); | |
// 得到 Field 对象的值 | |
System.out.println(field.get(clazz.newInstance())); | |
} |
补充:
Class 的 newInstance 方法
底层调用 无参的公共 的 Constructor 对象的 newInstance 方法创建该类的实例
5.2、设置 Field 对象的值
public static void main(String[] args) throws Exception {// 获得字节码对象 | |
Class clazz = Person.class; | |
// 得到指定的 Field | |
Field field = clazz.getField("name"); | |
// 给指定的 Field 对象设置值 | |
Object object = clazz.newInstance(); | |
field.set(object, "zhangsan"); | |
// 获取值 | |
System.out.println(field.get(object)); | |
} |
