1、Class对象
理解RTTI在Java中的工作原理,首先需要知道类型信息在运行时是如何表示的,这是由Class对象来完成的,它包含了与类有关的信息。Class对象就是用来创建所有“常规”对象的,Java使用Class对象来执行RTTI,即使你正在执行的是类似类型转换这样的操作。
每个类都会产生一个对应的Class对象,也就是保存在.class文件。所有类都是在对其第一次使用时,动态加载到JVM的,当程序创建一个对类的静态成员的引用时,就会加载这个类。Class对象仅在需要的时候才会加载,static初始化是在类加载时进行的。
类加载器首先会检查这个类的Class对象是否已被加载过,如果尚未加载,默认的类加载器就会根据类名查找对应的.class文件。
想在运行时使用类型信息,必须获取对象(比如类Base对象)的Class对象的引用,使用功能Class.forName(“Base”)可以实现该目的,或者使用base.class。注意,有一点很有趣,使用功能”.class”来创建Class对象的引用时,不会自动初始化该Class对象,使用forName()会自动初始化该Class对象。为了使用类而做的准备工作一般有以下3个步骤:
- 加载:由类加载器完成,找到对应的字节码,创建一个Class对象
- 链接:验证类中的字节码,为静态域分配空间
- 初始化:如果该类有超类,则对其初始化,执行静态初始化器和静态初始化块
@PostConstruct public void init() throws NoSuchFieldException { Field[] fields = this.getClass().getDeclaredFields(); Arrays.asList(fields) .forEach( field -> { Method[] methods = field.getType().getMethods(); Map<String, Method> methodMap = new ConcurrentHashMap<>(); Arrays.asList(methods) .forEach( method -> { methodMap.put(method.getName(), method); }); try { field.setAccessible(true); CLASS_METHODS.put(field.get(this), methodMap); } catch (IllegalAccessException e) { e.printStackTrace(); } }); }
上面的代码是填充一个map,map的每一个元素是这个类的成员变量,和这个变量的方法集合。
反射主要使用Class这个类的方法
补充资料: