JVM把描述类的数据从Class文件加载到内存,并对数据进行校验、转换解析和初始化,最终造成能够被虚拟机直接使用的Java类型。这就是类加载机制。java
Class文件结构中只有两种数据类型:数组
- 无符号数:属于基本的数据类型,以u一、u二、u四、u8分别表明一、二、四、8个字节的无符号数,能够用来描述数字、索引引用、数量值、或者按照UTF-8编码的字符串值;
- 表:由多个无符号数或者其余表做为数据项构成的复合数据类型,全部表都习惯以"_info"结尾;
- 因此,Class文件本质上就是一张表;
Class文件结构的内容组成:安全
- 魔数:每一个Class文件的前4个字节,JAVA语言魔数:CAFEBABE,16进制数,惟一左右就是确认文件是不是一个被虚拟机接受的Class文件;
- Class文件版本:Class文件第五、6个字节是次版本号,七、8个字节是主版本号;
- 常量池:从第9个字节开始,主要存放两大类常量:(1)字面量;(2)符号引用;
- 访问标志:紧跟常量池以后的两个字节表明访问标志,表明类或接口的层次信息;
- 类索引、父类索引与接口索引集合:紧跟访问标志以后的就是类索引、父类索引与接口索引,都是u2类型数据,Class文件中由这三项数据来肯定类的继承关系;
- 字段表集合:紧接着就是字段表集合,用于描述接口或类中声明的变量;
- 方法表集合:紧接着就是方法表集合,用于描述接口或类中声明的方法;
- 建立类的实例:使用new关键字实例化对象;
- 访问类的静态变量:getstatic或putstatic,读取或设置一个类的静态变量(不包括被final修饰的静态变量);
- 访问类的静态方法:invokestatic调用一个类的静态方法;
- 使用java.lang.reflect进行反射调用:如,Class.forName("xxxxx");
- 子类初始化时,会先初始化父类;
- 经过子类引用父类的静态字段,不会致使子类初始化,只会初始化父类;
- 经过数组定义来引用类,不会触发类的初始化;
- 常量在编译阶段会存入调用类的常量池中,本质上并无直接引用到定义常量的类,由于不会触发定义常量的类初始化;
- 经过类的全限定名称获取定义此类的二进制流;
- 将二进制流静态结构转化为方法区的运行时数据结构;
- 在内存中生成一个表明这个类的java.lang.Class对象。对于HotSpot虚拟机,Class对象是存放在方法区里的;
主要做用就是确保Class文件的字节流中包含的信息符合当前虚拟机的要求,而且不会危害虚拟机自身安全;数据结构
- 文件格式验证:对检查格式、版本;
- 元数据验证:对字节码进行语义分析;
- 字节码验证;
- 符号引用验证;
- 正式为类变量分配内存,内存在方法区中进行分配,类变量是指static修饰的变量;
- 设置类变量的初始值,这个初始值一般状况下是数据类型的零值,如:public static int value = 123;初始值是0,不是123,赋值123动做会在初始化阶段才会执行;
将Class文件的常量池内的符号引用替换为直接引用;函数
真正开始执行类中定义的Java程序代码,初始化过程就是执行类构造器()方法的过程;编码
- ()方法会自动收集类中全部类变量(static)的赋值动做和静态代码块,编译器收集顺序由代码出现顺序决定;
- ()方法与类实例构造函数不一样,虚拟机会保证父类的()执行完毕后再执行子类的()方法;
- 根类加载器(Bootstrap ClassLoader):其负责加载Java的核心类,好比String、System这些类;
- 拓展类加载器(Extension ClassLoader):其负责加载JRE的拓展类库;
- 系统类加载器(System ClassLoader):其负责加载CLASSPATH环境变量所指定的JAR包和类路径;
- 用户类加载器:用户自定义的加载器,以类加载器为父类
双亲委派机制:若是一个类加载器在接到加载类的请求时,它首先不会本身尝试去加载这个类,而是把这个请求任务委托给父类加载器去完成,依次递归,若是父类加载器能够完成类加载任务,就成功返回;只有父类加载器没法完成此加载任务时,才本身去加载。cdn
Java类随着加载它的类加载器一块儿具有了一种带有优先级的层次关系。好比,Java中的Object类,它存放在rt.jar之中,不管哪个类加载器要加载这个类,最终都是委派给处于模型最顶端的启动类加载器进行加载,所以Object在各类类加载环境中都是同一个类。若是不采用双亲委派模型,那么由各个类加载器本身取加载的话,那么系统中会存在多种不一样的Object类。对象