第六章 类文件结构
6.3 Class类文件的结构
Class文件前4个字节是魔数0xCAFEBABE,用来确定这个文件是否可以被虚拟机接受(后缀可以随便改,不安全)
5-8是版本号,其中5-6是次版本号,7-8是主版本号
接着是常量池
常量池结束,紧跟着2个字节是访问标志,用于识别类或接口的访问信息
类索引和父类索引,用来确定类的继承关系
字段表,用于描述接口或类中声明的变量(不包括局部变量)
方法表集合
属性表集合
第七章 虚拟机类加载机制
7.1 概述
Java虚拟机把描述类的数据从Class文件加载到内存,经过校验,转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型。
7.2 类加载的时机
加载--验证--准备--解析--初始化--使用--卸载
类初始化的六种情况:
1)遇到new、getstatic、putstatic、invokestatic时,如果类型没有初始化,需要触发初始化
2)使用反射调用的时候,如果类型没有初始化,需要触发初始化
3)初始化子类的时候,如果父类没有初始化,要先初始化父类
4)虚拟机启动时,一定要指定一个要执行的主类,这个主类先初始化
5)JDK7。。。
6)JDK8。。。
注1:对于静态字段,只有定义它的类才会被初始化,通过其子类引用父类的静态字段,不会触发子类初始化。
注2:通过数组定义来引用类,不会引起类的初始化
Person[] p = new Person();//不会初始化Person类
注3:常量被引用,不会引起类的初始化
7.4 类加载器
比较两个类是否相等,必须是两个类由同一个类加载器加载才有比较的意义。
即使两个类来源于同一个class文件,被同一个虚拟机加载,但被两个不同的类加载器加载,这两个类一定不相等
双亲委派模型:
1.工作过程:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载类,而是把这个请求委派给父类去完成,每一个层次的类加载器都是如此,因此所有的类加载请求最终都会到达最顶层的启动类加载器里,只有当父类无法加载请求时,才会由子加载器尝试加载。
2.好处:Java中的类和它的类加载器具备了一种优先级的层次关系。比如Object类,无论哪一个类加载器要加载这个类,都要把请求最终委派到启动类加载器进行加载,这样,Object类保证在各种类加载器环境中都是同一个类。反之,如果没有双亲委派模型,各个类加载器中的类都不一样,系统就乱了。
第八章 虚拟机字节码执行引擎
在不同的虚拟机实现中,执行引擎在执行字节码的时候,会用解释执行(通过解释器执行)和编译执行(通过即时编译器产生本地代码执行)两种选择,也可能两者兼备,有的还包括不同级别的即时编译器。
解释执行:源码执行一句,运行一句,每次加载都要执行;
编译执行:把代码一次全部编译成本地代码再执行,只加载一次;
封装、继承、多态
多态:字段没有多态性、方法才有
例如:Father son = new Son();
son.money();//这里打印的是父类的字段值
son.say();//这里调用的是子类的方法
第九章 类加载集执行子系统的案例与实战