学习资料来自java
深刻理解JVM虚拟机segmentfault
http://www.javashuo.com/article/p-msqwsymi-a.html数组
Java虚拟机的指令由一个字节长度的,表明着某种特定操做含义的数字(简称操做码,Opcode)以及跟随其后的零至多个表明此操做所需参数(称做操做数,Operands)来构成的。Java虚拟机采用面向操做数栈而不是寄存器(Android的Dalvik虚拟机则是依靠寄存器)架构的,因此大多数的指令都不包含操做数,只有一个操做码。post
因为Java虚拟机操做码的长度限制在了一个字节(0-255),意味着指令集的操做码总数不可能超过256条。学习
不考虑异常处理的话,Java虚拟机的解释器可使用下面的伪代码当作最基本的执行模型来理解:ui
do{ 自动计算PC寄存器的值+1; 根据PC寄存器的指示位置,从字节码流中取出操做码; if(字节码存在操做数) { 从字节流中取出操做数; } 执行操做码所定义的操做; }while(字节码长度>0)
在Java虚拟机的指令集中,大多数的指令都包含了其操做所对应的数据类型信息。如iload,fload指令用于从局部变量表中加载int,float型数据到操做数栈中。lua
对于大部分与数据类型相关的字节码指令,它们的操做码助记符中都有特殊的字符来代表专门为哪一种数据类型服务:i表明对int类型的数据操做,l表明long,s表明short,b表明byte,c表明char,f表明float,d表明double,a表明reference。 也有一些指令的助记符中没有明确地指明操做类型的字母,如arraylength指令,它没有表明数据类型的特殊字符,但操做数永远只能是一个数组类型的对象。 还有另一些指令,如无条件跳转指令goto则是与数据类型无关的。.net
Java虚拟机指令所支持的数据类型以下:
加载和存储指令用于将数据在栈帧中的局部变量表和操做数栈之间来回传输,指令内容以下:
记录常量指令的区别:
当int取值-1~5时,JVM采用iconst指令将常量压入栈中。
当int取值-128~127时,JVM采用bipush指令将常量压入栈中。
当int取值-32768~32767时,JVM采用sipush指令将常量压入栈中。
当int取值-2147483648~2147483647时,JVM采用ldc指令将常量压入栈中。
存储数据的操做数栈和局部变量表主要就是由加载和存储指令进行操做,除此以外,还有少许指令,如访问对象的字段或者数组元素的指令也会向操做数栈传输数据。
上述的_<n>表述一组数据,如iload_<n>,它表明了 iload_0、iload_一、iload_2 和 iload_3 这几条指令。iload_0的语义与操做数栈数为0时的iload指令语义彻底一致。
运算指令用于对两个操做数栈上的值进行某种特定特定运算,并把结果从新存储到操做栈顶。对于运算指令而言能够分红两种:** 对整形数据进行运算的指令与对浮点型数据进行运算的指令。 **不管哪一种运算指令,都是用Java虚拟机的数据类型,因为没有直接支持byte,short,char和boolean的运算指令,对于此类的运算,使用操做int类型的指令代替。运算指令以下:
类型转换指令可让两种不一样的数值类型进行互相转换。Java虚拟机直接支持如下数值类型的宽化类型转换(即小范围到大范围的转换):
对于窄话类型转换,须要显示使用转换指令完成,指令包括:i2b,i2c,i2s,l2i,f2i,f2l,d2i,d2l,d2f。对于窄化类型转换结果会致使转换结果产生不一样的正负号,精度丢失的状况,好比float转int。
虽然类实例和数组都是对象,可是Java虚拟机对于这两种的建立和操做使用了不一样的字节码指令,对象建立后,就能够经过对象访问指令获取对象实例或者数组实例中的字段或者数组元素。对应指令以下:
Java虚拟机提供了一下直接操做操做数栈的指令,包括:
控制转移指令就是在有条件或者无条件修改pc寄存器的值,即代码中的if,else,switch等关键字,指令以下:
方法返回指令是根据返回值的类型区分的,包括ireturn(返回值是boolean,byte,char,short和 int),lreturn,freturn,drturn和areturn,另一个return供void方法,实例初始化方法,类和接口的类初始化i方法使用。
在Java程序中显式抛出异常的操做(throw语句)都有athrow 指令来实现,除了用throw 语句显示抛出异常状况外,Java虚拟机规范还规定了许多运行时异常会在其余Java虚拟机指令检测到异常情况时自动抛出。 在Java虚拟机中,处理异常不是由字节码指令来实现的,而是采用异常表来完成的。
方法级的同步是隐式的,无需经过字节码指令来控制,它实如今方法调用和返回操做中。虚拟机从方法常量池中的方法标结构中的 ACC_SYNCHRONIZED标志区分是不是同步方法。方法调用时,调用指令会检查该标志是否被设置,若设置,执行线程先成功持有管程,而后才能执行方法,最后方法完成释放管程。
若是在同步方法执行期间,方法抛了异常,而且在内部没法处理异常,这个同步方法所持有的管程将会在异常抛到外部时自动释放。
同步一段指令集序列,一般由synchronized块标示,JVM指令集中有monitorenter和monitorexit来支持synchronized语义。 结构化锁定是指方法调用期间每个monitor退出都与前面monitor进入相匹配的情形。
总结:
这里只是学习了一下理论的知识,弄明白了对应指令作的对应的事情,经过javap反编译出来的文件中就包含了对应的指令:
public int testInt(java.lang.Object); descriptor: (Ljava/lang/Object;)I flags: ACC_PUBLIC Code: stack=2, locals=7, args_size=2 0: aload_1 1: dup 2: astore_2 3: monitorenter 4: bipush 11 6: istore_3 7: bipush 22 9: istore 4 11: aload_1 12: invokevirtual #12 // Method java/lang/Object.hashCod :()I 15: iconst_2 16: irem 17: ifne 27 20: bipush 33 22: istore_3 23: bipush 44 25: istore 4 27: iload_3 28: iload 4 30: iadd 31: istore 5 33: iload 5 35: aload_2 36: monitorexit 37: ireturn 38: astore_3 39: iconst_0 40: aload_2 41: monitorexit 42: ireturn 43: astore 6 45: aload_2 46: monitorexit 47: aload 6 49: athrow
在以后的学习中,须要应用的对应的分析当中,明白上述的执行过程。