与JVM做朋友系列(3)又见Class字节码
当一个人心中有更高的山峰去攀登,就不会在意脚下的泥沼。
前提概要
至此已经更新了三部分了,我们“不忘初心,方得始终”,回归了我们第一章课题的Class字节码,主要是考虑到如果总是一味的去了解class字节码,大家可能会枯燥到蒙圈啊,所以我们用类加载器的章节做一下过度,好了,接下来进入我们的正题,把Class字节码的学习贯彻到底吧!拥抱我们的“好朋友”->
字节码指令
字节码指令由一个 字节长 度大小并且代表着某种特定操作含义的数字(称为操作码,Opcode)以及跟随其后的零至多个代表此操作所需参数(称为操作数,Operands)而构成。由于限制了操作码的长度为一个字节(即 0~255),这意味着指令集的操作码总数不可能超过 256 条。
大部分的指令都没有支持整数类型
加载和存储指令
将一个局部变量加载到操作栈 :
iload、iload_ 、lload、lload_ 、fload、fload_ 、dload、dload_ 、aload、aload_ 。
将一个数值从操作数栈存储到局部变量表:
istore、istore_ 、lstore、lstore_ 、fstore、fstore_ 、dstore、dstore_ 、astore、astore_ 。
将一个常量加载到操作数栈:
bipush、sipush、ldc、ldc_w、ldc2_w、aconst_null、iconst_m1、iconst_、lconst_ 、fconst_ 、dconst_ 。
扩充局部变量表的访问索引的指令:
wide
运算或算术指令
用于对两个操作数栈上的值进行某种特定运算,并把结果重新存入到操作栈顶。
加法指令:
减法指令:
乘法指令:
类型转换的指令
可以将两种不同的数值类型进行相互转换,
int 类型到 long、float 或者 double 类型。 long 类型到 float、double 类型。 float 类型到 double 类型。
处理窄化类型转换(Narrowing Numeric Conversions)时,必须显式地使用转换指令来完成,这些转换指令包括:
创建类实例的指令:
new
创建数组的指令:
newarray、anewarray、multianewarray。
访问字段指令:
getfield、putfield、getstatic、putstatic。
数组存取相关指令
把一个数组元素加载到操作数栈的指令:
baload、caload、saload、iaload、laload、faload、daload、aaload。
将一个操作数栈的值存储到数组元素中的指令:
bastore、castore、sastore、iastore、fastore、dastore、aastore。
取数组长度的指令:
arraylength
检查类实例类型的指令:
instanceof、checkcast。
操作数栈管理指令
如同操作一个普通数据结构中的堆栈那样,Java 虚拟机提供了一些用于直接操作操作数栈的指令,包括:将操作数栈的栈顶一个或两个元素出栈:pop、pop2。复制栈顶一个或两个数值并将复制值或双份的复制值重新压入栈顶:dup、dup2、dup_x1、dup2_x1、dup_x2、dup2_x2。
将栈最顶端的两个数值互换:swap。
控制转移指令
控制转移指令可以让 Java 虚拟机有条件或无条件地从指定的位置指令而不是控制转移指令的下一条指令继续执行程序,从概念模型上理解,可以认为控制转移指令就是在有条件或无条件地修改 PC 寄存器的值。控制转移指令如下。
条件分支:
ifeq、iflt、ifle、ifne、ifgt、ifge、ifnull、ifnonnull、if_icmpeq、if_icmpne、if_icmplt、if_icmpgt、if_icmple、if_icmpge、if_acmpeq 和 if_acmpne。
复合条件分支:
tableswitch、lookupswitch。
无条件分支:
goto、goto_w、jsr、jsr_w、ret。
方法调用指令
invokevirtual
指令用于调用对象的实例方法, 根据对象的实际类型进行分派(虚方法分派),这也是 Java 语言中最常见的方法分派方式。invokeinterface
指令用于调用接口方法 ,它会在运行时搜索一个实现了这个接口方法的对象,找出适合的方法进行调用。invokespecial 指令用于调用一些需要特殊处理的实例方法,包括实例初始化方法、私有方法和父类方法。 invokestatic 指令用于调用类方法(static 方法)。
invokedynamic 指令用于在运行时动态解析出调用点限定符所引用的方法,并执行该方法,前面 4 条调用指令的分派逻辑都固化在 Java 虚拟机内部,而 invokedynamic 指令的分派逻辑是由用户所设定的引导方法决定的。方法调用指令与数据类型无关。
方法返回指令
是根据返回值的类型区分的,包括 ireturn(当返回值是 boolean、byte、char、short 和 int类型时使用)、lreturn、freturn、dreturn 和 areturn,另外还有一条 return 指令供声明为void 的方法、实例初始化方法以及类和接口的类初始化方法使用。
异常处理指令
在Java 程序中显式抛出异常的操作(throw 语句)都由 athrow 指令来实现
同步指令
解析
在Java 语言中符合“编译期可知,运行期不可变”这个要求的方法,主要包括静态方法和私有方法两大类,前者与类型直接关联,后者在外部不可被访问,这两种方法各自的特点决定了它们都不可能通过继承或别的方式重写其他版本,
静态分派
多见于方法的重载。

“Human”称为变量的静态类型(Static Type),或者叫做的外观类型(ApparentType),后面的“Man”则称为变量的实际类型(Actual Type),静态类型和实际类型在程序中都可以发生一些变化,区别是静态类型的变化仅仅在使用时发生,变量本身的静态类型不会被改变,并且最终的静态类型是在编译期可知的;
动态分派
静态类型同样都是 Human 的两个变量 man 和 woman 在调用 sayHello()方法时执行了不同的行为,并且变量 man 在两次调用中执行了不同的方法。导致这个现象的原因很明显,是这两个变量的实际类型不同。在实现上,最常用的手段就是为类在方法区中建立一个虚方法表。虚方法表中存放着各个方法的实际入口地址。如果某个方法在子类中没有被重写,那子类的虚方法表里面的地址入口和父类相同方法的地址入口是一致的,都指向父类的实现入口。如果子类中重写了这个方法,子类方法表中的地址将会替换为指向子类实现版本的入口地址,Son重写了来自 Father 的全部方法,因此 Son 的方法表没有指向 Father 类型数据的箭头。但是 Son 和 Father 都没有重写来自 Object 的方法,所以它们的方法表中所有从 Object 继承来的方法都指向了 Object 的数据类型。
基于栈的字节码解释执行引擎
Java编译器输出的指令流,
例子,分别使用这两种指令集计算“1+1”的结果,基于栈的指令集会是这样子的:
iconst_1
iconst_1
iadd
istore_0
两条 iconst_1 指令连续把两个常量 1 压入栈后,iadd 指令把栈顶的两个值出栈、相加,然后把结果放回栈顶,最后 istore_0 把栈顶的值放到局部变量表的第 0 个 Slot 中。
如果基于寄存器,那程序可能会是这个样子:
mov eax,1add eax,1mov 指令把 EAX 寄存器的值设为 1,然后 add 指令再把这个值加 1,结果就保存在 EAX寄存器里面。
