耐心看完下面的讲解你会受益匪浅~
1.先简单介绍一下java.lang.Class<T>:
T
- 由此 Class
对象建模的类的类型。
Class
类的实例表示正在运行的 Java 应用程序中的类和接口。
2.由Java代码在计算机中经历的三个阶段来引入反射:
(Java类编译为.class文件,从.class文件加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括,加载 ,验证 , 准备 , 解析 , 初始化 , 卸载 ,总共七个阶段。其中验证 ,准备 , 解析 统称为连接。)
2.1讲一下类加载器:
1.类加载过程
当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载,连接,初始化三步来实现对这个类进行初始化。
a.加载
就是指将class文件读入内存,并为之创建一个Class对象。
任何类被使用时系统都会建立一个Class对象
b.连接
验证 是否有正确的内部结构,并和其他类协调一致
准备 负责为类的静态成员分配内存,并设置默认初始化值
解析 将类的二进制数据中的符号引用替换为直接引用
(符号引用和直接引用可以看:
https://blog.csdn.net/qq_34402394/article/details/72793119)
c.初始化
还是比较复杂的
详见:https://www.cnblogs.com/jimxz/p/3974939.html
2.类加载器的功能
负责将.class文件加载到内在中,并为之生成对应的Class对象。
虽然我们不需要关心类加载机制,但是了解这个机制我们就能更好的理解程序的运行
3. 什么是反射?
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
要想解剖一个类,必须先要获取到该类的字节码文件对象。而解剖使用的就是Class类中的方法。所以先要获取到每一个字节码文件对应的Class类型的对象。
4.Class类
阅读API的Class类得知,Class 没有公共构造方法。Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的 defineClass 方法自动构造的
获取Class对象的三种方式:
//通过Object类中的getClass()方法
//多用于对象的获取字节码的方式
Person p = new Person();
Class c = p.getClass();
//方式二: 通过 类名.class 获取到字节码文件对象(任意数据类型都具备一个class静态属性,看上去要比第一种方式简单)。
// 多用于参数的传递
Class c2 = Person.class;
//方式三: 通过Class类中的方法(将类名作为字符串传递给Class类中的静态方法forName即可)。
//多用于配置文件,将类名定义在配置文件中。读取文件,加载类
Class c3 = Class.forName("包名.Person");
//结论:同一个字节码文件(*.class)在一次程序运行过程中,只会被加载一次,不论通过哪一种方式获取的Class对象都是同一个。
5. Class对象的获取功能:
1. 获取成员变量们
* Field[] getFields() :获取所有public修饰的成员变量
* Field getField(“属性名”) 获取指定名称的 public修饰的成员变量
* Field[] getDeclaredFields() 获取所有的成员变量,不考虑修饰符
* Field getDeclaredField(“属性名”)
通过方法,给指定对象的指定成员变量赋值或者获取值:
1.public void set(Object obj, Object value)
//在指定对象obj中,将此 Field 对象表示的成员变量设置为指定的新值
2.public Object get(Object obj)
//返回指定对象obj中,此 Field 对象表示的成员变量的值
2. 获取构造方法们
* Constructor<?>[] getConstructors()
* Constructor<T> getConstructor(String.class/int.class参数的数据类型的class文件)
* Constructor<T> getDeclaredConstructor(String.class/int.class参数的数据类型的class文件)
* Constructor<?>[] getDeclaredConstructors()
通过构造方法创建对象:
Object obj = con.newInstance(null);//无参构造
Object obj = con.newInstance("小明", 22, "哈尔滨");//有参构造
3. 获取成员方法们:
* Method[] getMethods()
* Method getMethod("方法名()",String.class/int.class参数的数据类型的class文件)
* Method[] getDeclaredMethods()
* Method getDeclaredMethod("方法名()",String.class/int.class参数的数据类型的class文件)
public Object invoke(Object obj, 该方法需要的参数)
//执行指定对象obj中,当前Method对象所代表的方法,方法要传入的参数通过args指定。
4. 获取全类名
* String getName()
5. 暴力反射(就是私有的属性、方法也能访问到)
AccessibleObject 类是 Field、Method 和 Constructor 对象的父类。它提供了将反射的对象标记为在使用时取消默认 Java 语言访问控制检查的能力。
对于公共成员、默认(打包)访问成员、受保护成员和私有成员,在分别使用 Field、Method 或 Constructor 对象来设置或获取字段、调用方法,或者创建和初始化类的新实例的时候,会执行访问检查。常用方法如下:
public void setAccessible(boolean flag) throws SecurityException
参数值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查。参数值为 false 则指示反射的对象应该实施 Java 语言访问检查。