反射(Reflect)
我在java注解文章中, 有提到 RetentionPolicy.RUNTIME, 主要是用在反射技术上. 顾名思义JAVA反射机制是在运行状态中的技术.反射就是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性; 并且能改变它的属性. 在Java中被视为动态语言的关键
现在我们思考一个问题:既然在运行状态中可以反射的Class类的信息,那么这些类的信息到底存在JVM的哪里?
在搞清楚上面之前, 我们先来简单了解一下JVM虚拟机. 如果想详细了解JVM虚拟机请关注我后面的文章.
JVM简单概括(除HostSpot JVM用户)
JVM 从线程角度来看, 分为线程私有和线程共享区,
其中线程共享区:分为堆,方法区
线程私有包含: 程序计数器,本地方法栈,虚拟机栈; 据说待考证, 在jdk8 之后,把本地方法栈和虚拟机栈合并了.
这里我们只简单的介绍线程共享区域里面的方法区,
方法区:是各个线程共享的内存区域,它主要是用于存储已被虚拟机加载的类信息,常量,静态变量,及时编译器编译后的代码数据.
Class类
Class类是一个Java基础类, 每当装载一个新的类型的时候,java虚拟机都会在java堆中创建一个对应于新类型的Class, 该实例就代表此类型,通过该Class实例我们就可以访问该类型的基本信息. 上面说到在方法区中存储被装载类的类型信息, 我们可以通过Class实例来访问这些信息.
下面是获取类信息的对应的少部分的API,其他的API可以查看JDK帮助文档:
1:getName() 获取这个类的全限定名
2:getSuperClass() 获取这个类型的直接父类的全限定名
3:isInterface() 判断这个类型是否是接口类型
4:getTypeParamters() 获取这个类型的范围修饰符, 本文后面会讲到Type
5:getInterfaces() 获取这个类型的超接口的全限定名的列表
6:getFields() 获取公有字段,包括父类
7,getDeclaredFields() 获取当前类所有的字段 不包括父类的公有字段
8,getMethods()获取公有方法,包括父类
9,getDeclaredMethods() 获取当前类所有的方法, 不包括父类的公有方法.
怎样获取到Class的对象?
获取Class对象API 有:
类名.Class
对象.getClass()
Class.forName("全限定名")
类.getClassLoader().loadClass("全限定名");
子类.class.getSuperClass()
包装类.class
这是带有注释的JDK8里面的loadClass()方法源码, 在源码中可以看出, 双亲委派机制,通过父对象获取到Class对象. 在Android中经常会遇到.
File:JDK1.8-->ClassLoader.java
protected Class> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) {
// First, check if the class has already been loaded
Class> c = findLoadedClass(name); //首先从缓存中获取class。
if (c == null) {
long t0 = System.nanoTime();
try {
if (parent != null) {//这里是典型的双亲委派机制, 如果父亲有就直接拿父亲的。 也叫坑爹机制
c = parent.loadClass(name, false);
} else {
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
// ClassNotFoundException t