深入理解JVM - 虚拟机字节码指令集
Java虚拟机的指令由一个字节长度的、代表着某种特定操作含义的数字(称为操作码,Opcode)以及跟随其后的零至多个代表此操作所需的参数(称为操作数,Operand)构成。因为只有一个字节的长度,所以指令总数不能超过256个。
在Java虚拟机的指令集中,大多数指令都包含其操作所对应的数据类型信息,如:i代表对int类型的数据操作,l代表long,s代表short,b代表byte,c代表char,f代表float,d代表double,a代表reference。
解释器的执行模型
Java虚拟机的解释器的执行模型:
do {自动计算PC寄存器的值加1;根据PC寄存器指示的位置,从字节码流中取出操作码;if (字节码存在操作数) 从字节码流中取出操作数;执行操作码所定义的操作;
} while (字节码流长度 > 0);
常量入栈指令
指令码 | 操作码(助记符) | 操作数 | 描述(栈指操作数栈) |
---|---|---|---|
0x01 | aconst_null | 将 null推送至栈顶 | |
0x02 | iconst_m1 | 将 -1(int)推送至栈顶 | |
0x03 | iconst_0 | 将 0(int)推送至栈顶 | |
0x04 | iconst_1 | 将 1(int)推送至栈顶 | |
0x05 | iconst_2 | 将 2(int)推送至栈顶 | |
0x06 | iconst_3 | 将 3(int)推送至栈顶 | |
0x07 | iconst_4 | 将 4(int)推送至栈顶 | |
0x08 | iconst_5 | 将 5(int)推送至栈顶 | |
0x09 | lconst_0 | 将 0(long)推送至栈顶 | |
0x0a | lconst_1 | 将 1(long)推送至栈顶 | |
0x0b | fconst_0 | 将 0(float)推送至栈顶 | |
0x0c | fconst_1 | 将 1(float)推送至栈顶 | |
0x0d | fconst_2 | 将 2(float)推送至栈顶 | |
0x0e | dconst_0 | 将 0(double)推送至栈顶 | |
0x0f | dconst_1 | 将 1(double)推送至栈顶 | |
0x10 | bipush | valuebyte | 将一个byte值带符号扩展成int推送至栈顶 |
0x11 | sipush |
valuebyte1 valuebyte2 |
将一个short值带符号扩展成int推送至栈顶 |
0x12 | ldc | indexbyte1 | 将int、float或String型常量值从常量池中推送至栈顶 |
0x13 | ldc_w |
indexbyte1 indexbyte2 |
将int、float或String型常量值从常量池中推送至栈顶(宽索引) |
0x14 | ldc2_w |
indexbyte1 indexbyte2 |
将long或double型常量值从常量池中推送至栈顶(宽索引) |
局部变量值转载到栈中指令
指令码 | 操作码(助记符) | 操作数 | 描述(栈指操作数栈) |
---|---|---|---|
0x15 | iload | indexbyte | 从局部变量indexbyte中装载int类型值推送至栈顶 |
0x16 | lload | indexbyte | 从局部变量indexbyte中装载long类型值推送至栈顶 |
0x17 | fload | indexbyte | 从局部变量indexbyte中装载float类型值推送至栈顶 |
0x18 | dload | indexbyte | 从局部变量indexbyte中装载double类型值推送至栈顶 |
0x19 | aload | indexbyte | 从局部变量indexbyte中装载引用类型值推送至栈顶 |
0x1a | iload_0 | 将第1个int型本地变量推送至栈顶 | |
0x1b | iload_1 | 将第2个int型本地变量推送至栈顶 | |
0x1c | iload_2 | 将第4个int型本地变量推送至栈顶 | |
0x1d | iload_3 | 将第4个int型本地变量推送至栈顶 | |
0x1e | lload_0 | 将第1个long型本地变量推送至栈顶 | |
0x1f | lload_1 | 将第2个long型本地变量推送至栈顶 | |
0x20 | lload_2 | 将第3个long型本地变量推送至栈顶 | |
0x21 | lload_3 | 将第4个long型本地变量推送至栈顶 | |
0x22 | fload_0 | 将第1个float型本地变量推送至栈顶 | |
0x23 | fload_1 | 将第2个float型本地变量推送至栈顶 | |
0x24 | fload_2 | 将第3个float型本地变量推送至栈顶 | |
0x25 | fload_3 | 将第4个float型本地变量推送至栈顶 | |
0x26 | dload_0 | 将第1个double型本地变量推送至栈顶 | |
0x27 | dload_1 | 将第2个double型本地变量推送至栈顶 | |
0x28 | dload_2 | 将第3个double型本地变量推送至栈顶 | |
0x29 | dload_3 | 将第4个double型本地变量推送至栈顶 | |
0x2a | aload_0 | 将第1个引用类型本地变量推送至栈顶 | |
0x2b | aload_1 | 将第2个引用类型本地变量推送至栈顶 | |
0x2c | aload_2 | 将第3个引用类型本地变量推送至栈顶 | |
0x2d | aload_3 | 将第4个引用类型本地变量推送至栈顶 | |
0x2e | iaload | 将int类型数组的索引值推送至栈顶 | |
0x2f | laload | 将long类型数组的索引值推送至栈顶 | |
0x30 | faload | 将float类型数组的索引值推送至栈顶 | |
0x31 | daload | 将double类型数组的索引值推送至栈顶 | |
0x32 | aaload | 将引用类型数组的索引值推送至栈顶 | |
0x33 | baload | 将boolean或byte类型数组的索引值推送至栈顶(先转换为int类型值,后压栈) | |
0x34 | caload | 将char类型数组的索引值推送至栈顶(先转换为int类型值,后压栈) | |
0x35 | saload | 将short类型数组的索引值推送至栈顶(先转换为int类型值,后压栈) |
将栈顶值保存到局部变量中指令
指令码 | 操作码(助记符) | 操作数 | 描述(栈指操作数栈) |
---|---|---|---|
0x36 | (wide)istore | indexbyte | 将栈顶int类型值保存到局部变量indexbyte中 |
0x37 | (wide)lstore | indexbyte | 将栈顶long类型值保存到局部变量indexbyte中 |
0x38 | (wide)fstore | indexbyte | 将栈顶float类型值保存到局部变量indexbyte中 |
0x39 | (wide)dstore | indexbyte | 将栈顶double类型值保存到局部变量indexbyte中 |
0x3a | (wide)astore | indexbyte | 将栈顶引用类型值保存到局部变量indexbyte中 |
0x3b | istore_0 | 将栈顶int类型值保存到局部变量0中 | |
0x3c | istore_1 | 将栈顶int类型值保存到局部变量1中 | |
0x3d | istore_2 | 将栈顶int类型值保存到局部变量2中 | |
0x3e | istore_3 | 将栈顶int类型值保存到局部变量3中 | |
0x3f | lstore_0 | 将栈顶long类型值保存到局部变量0中 | |
0x40 | lstore_1 | 将栈顶long类型值保存到局部变量1中 | |
0x41 | lstore_2 | 将栈顶long类型值保存到局部变量2中 | |
0x42 | lstroe_3 | 将栈顶long类型值保存到局部变量3中 | |
0x43 | fstore_0 | 将栈顶float类型值保存到局部变量0中 | |
0x44 | fstore_1 | 将栈顶float类型值保存到局部变量1中 | |
0x45 | fstore_2 | 将栈顶float类型值保存到局部变量2中 | |
0x46 | fstore_3 | 将栈顶float类型值保存到局部变量3中 | |
0x47 | dstore_0 | 将栈顶double类型值保存到局部变量0中 | |
0x48 | dstore_1 | 将栈顶double类型值保存到局部变量1中 | |
0x49 | dstore_2 | 将栈顶double类型值保存到局部变量2中 | |
0x4a | dstore_3 | 将栈顶double类型值保存到局部变量3中 | |
0x4b | astroe_0 | 将栈顶引用类型值保存到局部变量0中 | |
0x4c | astore_1 | 将栈顶引用类型值保存到局部变量1中 | |
0x4d | astore_2 | 将栈顶引用类型值保存到局部变量2中 | |
0x4e | astore_3 | 将栈顶引用类型值保存到局部变量3中 | |
0x4f | iastore | 将栈顶int类型值保存到指定int类型数组的指定索引位 | |
0x50 | lastore | 将栈顶long类型值保存到指定long类型数组的指定索引位 | |
0x51 | fastore | 将栈顶float类型值保存到指定float类型数组的指定索引位 | |
0x52 | dastore | 将栈顶double类型值保存到指定double类型数组的指定索引位 | |
0x53 | aastore | 将栈顶引用类型值保存到指定引用类型数组的指定索引位 | |
0x54 | bastroe | 将栈顶boolean类型值或byte类型值保存到指定boolean类型数组或byte类型数组的指定索引位 | |
0x55 | castore | 将栈顶char类型值保存到指定char类型数组的指定索引位 | |
0x56 | sastore | 将栈顶short类型值保存到指定short类型数组的指定索引位 |
通用(无类型)栈操作指令
指令码 | 操作码(助记符) | 操作数 | 描述(栈指操作数栈) |
---|---|---|---|
0x00 | nop | 空操作 | |
0x57 | pop | 从栈顶弹出一个字长的数据(不是long和double) | |
0x58 | pop2 | 从栈顶弹出两个字长的数据(一个long或double,或者是两个单字节长度数值) | |
0x59 | dup | 复制栈顶一个字长的数据,将复制后的数据压入栈顶 | |
0x5a | dup_x1 | 复制栈顶一个字长的数据,弹出栈顶两个字长数据,先将复制后的数据压入栈顶,再将弹出的两个字长数据压入栈顶 | |
0x5b | dup_x2 | 复制栈顶一个字长的数据,弹出栈顶三个字长的数据,将复制后的数据压入栈顶,再将弹出的三个字长的数据压入栈顶 | |
0x5c | dup2 | 复制栈顶两个字长的数据,将复制后的两个字长的数据压入栈顶 | |
0x5d | dup2_x1 | dup_x1 指令的双倍版 | |
0x5e | dup2_x2 | dup_x2 指令的双倍版 | |
0x5f | swap | 将栈最顶端的两个数值互换(数值不能是long或double) |
整数和浮动点数运算
指令码 | 操作码(助记符) | 操作数 | 描述(栈指操作数栈) |
---|---|---|---|
0x60 | iadd | 将栈顶两int类型数相加,并将结果压入栈顶 | |
0x61 | ladd | 将栈顶两long类型数相加,并将结果压入栈顶 | |
0x62 | fadd | 将栈顶两float类型数相加,并将结果压入栈顶 | |
0x63 | dadd | 将栈顶两double类型数相加,并将结果压入栈顶 | |
0x64 | isub | 将栈顶两int类型数相减,并将结果压入栈顶 | |
0x65 | lsub | 将栈顶两long类型数相减,并将结果压入栈顶 | |
0x66 | fsub | 将栈顶两float类型数相减,并将结果压入栈顶 | |
0x67 | dsub | 将栈顶两double类型数相减,并将结果压入栈顶 | |
0x68 | imul | 将栈顶两int类型数相乘,并将结果压入栈顶 | |
0x69 | lmul | 将栈顶两long类型数相乘,并将结果压入栈顶 | |
0x6a | fmul | 将栈顶两float类型数相乘,并将结果压入栈顶 | |
0x6b | dmul | 将栈顶两double类型数相乘,并将结果压入栈顶 | |
0x6c | idiv | 将栈顶两int类型数相除,并将结果压入栈顶 | |
0x6d | ldiv | 将栈顶两long类型数相除,并将结果压入栈顶 | |
0x6e | fdiv | 将栈顶两float类型数相除,并将结果压入栈顶 | |
0x6f | ddiv | 将栈顶两double类型数相除,并将结果压入栈顶 | |
0x70 | irem | 将栈顶两int类型数取模,并将结果压入栈顶 | |
0x71 | lrem | 将栈顶两long类型数取模,并将结果压入栈顶 | |
0x72 | frem | 将栈顶两float类型数取模,并将结果压入栈顶 | |
0x73 | drem | 将栈顶两double类型数取模,并将结果压入栈顶 | |
0x74 | ineg | 将栈顶int类型值取负,并将结果压入栈顶 | |
0x75 | lneg | 将栈顶long类型值取负,并将结果压入栈顶 | |
0x76 | fneg | 将栈顶float类型值取反,并将结果压入栈顶 | |
0x77 | dneg | 将栈顶double类型值取负,并将结果压入栈顶 | |
0x84 | (wide)iinc |
indexbyte constbyte |
将整数值constbyte加到indexbyte指定的int类型的局部变量中(i++,i–,i+=2;) |
逻辑运算 - 移位运算
指令码 | 操作码(助记符) | 操作数栈 | 描述(栈指操作数栈) |
---|---|---|---|
0x78 | ishl | … , a , n | (a << n) 左移int类型值,并将结果压入栈顶 |
0x79 | lshl | … , a , n | (a << n) 左移long类型值,并将结果压入栈顶 |
0x7a | ishr | … , a , n | (a >> n) 算术右移int类型值,并将结果压入栈顶 |
0x7b | lshr | … , a , n | (a >> n) 算术右移long类型值,并将结果压入栈顶 |
0x7c | iushr | … , a , n | (a >>> n) 逻辑右移int类型值,并将结果压入栈顶 |
0x7d | lushr | … , a , n | (a >>> n) 逻辑右移long类型值,并将结果压入栈顶 |
逻辑运算 - 位运算
指令码 | 操作码(助记符) | 操作数栈 | 描述(栈指操作数栈) |
---|---|---|---|
0x7e | iand | … , a , n | (a & b) 对int类型按位与运算,并将结果压入栈顶 |
0x7f | land | … , a , n | (a & b) 对long类型的按位与运算,并将结果压入栈顶 |
0x80 | ior | … , a , n | (a | b) 对int类型的按位或运算,并将结果压入栈顶 |
0x81 | lor | … , a , n | (a | b) 对long类型的按位或运算,并将结果压入栈顶 |
0x82 | ixor | … , a , n | (a ^ b) 对int类型的按位异或运算,并将结果压入栈顶 |
0x83 | lxor | … , a , n | (a ^ b) 对long类型的按位异或运算,并将结果压入栈顶 |
类型转换指令
指令码 | 操作码(助记符) | 操作数栈 | 描述(栈指操作数栈) |
---|---|---|---|
0x85 | i2l | … , a | (long) a, 将栈顶int类型值转换为long类型值,并将结果压入栈顶 |
0x86 | i2f | … , a | (float) a, 将栈顶int类型值转换为float类型值,并将结果压入栈顶 |
0x87 | i2d | … , a | (double) a, 将栈顶int类型值转换为double类型值,并将结果压入栈顶 |
0x88 | l2i | … , a | (int) a, 将栈顶long类型值转换为int类型值,并将结果压入栈顶 |
0x89 | l2f | … , a | (float) a, 将栈顶long类型值转换为float类型值,并将结果压入栈顶 |
0x8a | l2d | … , a | (double) a, 将栈顶long类型值转换double类型值,并将结果压入栈顶 |
0x8b | f2i | … , a | (int) a, 将栈顶float类型值转换为int类型值,并将结果压入栈顶 |
0x8c | f2l | … , a | (long) a, 将栈顶float类型值转换为long类型值,并将结果压入栈顶 |
0x8d | f2d | … , a | (double) a, 将栈顶float类型值转换为double类型值,并将结果压入栈顶 |
0x8e | d2i | … , i | (double) a, 将栈顶double类型值转换为int类型值,并将结果压入栈顶 |
0x8f | d2l | … , a | (long) a, 将栈顶double类型值转换为long类型值,并将结果压入栈顶 |
0x90 | d2f | … , a | (float) a, 将栈顶double类型值转换为float类型值,并将结果压入栈顶 |
0x91 | i2b | … , a | (byte) a, 将栈顶int类型值截断成byte类型,后带符号扩展成int类型值压入栈顶 |
0x92 | i2c | … , a | (char) a, 将栈顶int类型值截断成char类型值,后带符号扩展成int类型值压入栈顶 |
0x93 | i2s | … , a | (short) a, 将栈顶int类型值截断成short类型值,后带符号扩展成int类型值压入栈顶 |
控制流指令 - 比较指令
指令码 | 操作码(助记符) | 操作数栈 | 描述(栈指操作数栈) |
---|---|---|---|
0x94 | lcmp | … , a , b | 比较栈顶两long类型值,前者大,1入栈;相等,0入栈;后者大,-1入栈 [a == b ? 0 : (a < b ? -1 : 1)] |
0x95 | fcmpl | … , a , b | 比较栈顶两float类型值,前者大,1入栈;相等,0入栈;后者大,-1入栈;有NaN存在,-1入栈 [a == b ? 0 : (a < b ? -1 : 1)] |
0x96 | fcmpg | … , a , b | 比较栈顶两float类型值,前者大,1入栈;相等,0入栈;后者大,-1入栈;有NaN存在,-1入栈 [a == b ? 0 : (a < b ? -1 : 1)] |
0x97 | dcmpl | … , a , b | 比较栈顶两double类型值,前者大,1入栈;相等,0入栈;后者大,-1入栈;有NaN存在,-1入栈 [a == b ? 0 : (a < b ? -1 : 1)] |
0x98 | dcmpg | … , a , b | 比较栈顶两double类型值,前者大,1入栈;相等,0入栈;后者大,-1入栈;有NaN存在,-1入栈 [a == b ? 0 : (a < b ? -1 : 1)] |
控制流指令 - 条件跳转指令
指令码 | 操作码(助记符) | 栈操作之前 | 栈操作之后 | 描述(栈指操作数栈) |
---|---|---|---|---|
0x99 | ifeq | … , i | … | 若栈顶int类型值为0则跳转 (jump if i == 0) |
0x9a | ifne | … , i | … | 若栈顶int类型值不为0则跳转 (jump if i != 0) |
0x9b | iflt | … , i | … | 若栈顶int类型值小于0则跳转 (jump if i < 0) |
0x9c | ifge | … , i | … | 若栈顶int类型值大于等于0则跳转 (jump if i >= 0) |
0x9d | ifgt | … , i | … | 若栈顶int类型值大于0则跳转 (jump if i > 0) |
0x9e | ifle | … , i | … | 若栈顶int类型值小于等于0则跳转 (jump if i <= 0) |
0x9f | if_icmpeq | … , i , j | … | 若栈顶两int类型值相等则跳转 (jump if i == j) |
0xa0 | if_icmpne | … , i , j | … | 若栈顶两int类型值不相等则跳转 (jump if i != j) |
0xa1 | if_icmplt | … , i , j | … | 若栈顶两int类型值前小于后则跳转 (jump if i < j) |
0xa4 | if_icmple | … , i , j | … | 若栈顶两int类型值前小于等于后则跳转 (jump if i <= j) |
0xa3 | if_icmpgt | … , i , j | … | 若栈顶两int类型值前大于后则跳转 (jump if i > j) |
0xa2 | if_icmpge | … , i , j | … | 若栈顶两int类型值前大于等于后则跳转 (jump if i >= j) |
0xa5 | if_acmpeq | … , o , p | … | 若栈顶两引用类型值相等则跳转 (jump if o == p) |
0xa6 | if_acmpne | … , o , p | … | 若栈顶两引用类型值不相等则跳转 (jump if o != p) |
0xc6 | ifnull | … , o | … | 若栈顶引用值为null则跳转 (jump if o == null) |
0xc7 | ifnonnull | … , o | … | 若栈顶引用值不为null则跳转 (jump if o != null) |
控制流指令 - 无条件跳转指令
指令码 | 操作码(助记符) | 操作数 | 描述(栈指操作数栈) |
---|---|---|---|
0xa7 | goto |
branchbyte1 branchbyte2 |
无条件跳转到指定位置 |
0xc8 | goto_w |
branchbyte1 branchbyte2 branchbyte3 branchbyte4 |
无条件跳转到指定位置(宽索引) |
控制流指令 - 表跳转指令
指令码 | 操作码(助记符) | 操作数 | 描述(栈指操作数栈) |
---|---|---|---|
0xaa | tableswitch |
<0 - 3bytepad> defaultbyte1 defaultbyte2 defaultbyte3 defaultbyte4 lowbyte1 lowbyte2 lowbyte3 lowbyte4 highbyte1 highbyte2 highbyte3 highbyte4 jump offsets… |
通过索引访问跳转表,并跳转 jump always |
0xab | lookupswitch |
<0 - 3bytepad> defaultbyte1 defaultbyte2 defaultbyte3 defaultbyte4 npairs1 npairs2 npairs3 npairs4 match offsets |
通过键值访问跳转表,并跳转 jump always |
控制流指令 - 异常和finally
指令码 | 操作码(助记符) | 操作数 | 描述(栈指操作数栈) |
---|---|---|---|
0xbf | athrow | 抛出异常 | |
0xa8 | jsr |
branchbyte1 branchbyte2 |
跳转到子例程序 |
0xc9 | jsr_w |
branchbyte1 branchbyte2 branchbyte3 branchbyte4 |
跳转到子例程序(宽索引) |
0xa9 | (wide)ret | indexbyte | 返回子例程序 |
对象操作指令
指令码 | 操作码(助记符) | 操作数 | 描述(栈指操作数栈) |
---|---|---|---|
0xbb | new |
indexbyte1 indexbyte2 |
创建新的对象实例,并将其引用压入栈顶 |
0xc0 | checkcast |
indexbyte1 indexbyte |
类型强转,如果该检查未通过将会抛出ClassCastException异常 |
0xc1 | instanceof |
indexbyte1 indexbyte2 |
检查对象是否是指定的类的实例。如果是,1进栈;否则,0进栈 |
0xb2 | getstatic |
indexbyte1 indexbyte2 |
获取静态字段的值,并将其引用压入栈顶 |
0xb3 | putstatic |
indexbyte1 indexbyte2 |
给静态字段赋值 |
0xb4 | getfield |
indexbyte1 indexbyte2 |
获取对象字段的值,并将其引用压入栈顶 |
0xb5 | putfield |
indexbyte1 indexbyte2 |
给对象字段赋值 |
数组操作指令
指令码 | 操作码(助记符) | 操作数 | 描述(栈指操作数栈) |
---|---|---|---|
0xbc | newarray | atype | 创建type类型的数组 |
0xbd | anewarray |
indexbyte1 indexbyte2 |
创建引用类型的数组 |
0xbe | arraylength | 获取一维数组的长度 | |
0xc5 | multianewarray |
indexbyte1 indexbyte2 dimension |
创建dimension维度的数组 |
方法调用指令
指令码 | 操作码(助记符) | 操作数 | 描述(栈指操作数栈) |
---|---|---|---|
0xb7 | invokespecial |
indexbyte1 indexbyte2 |
编译时方法绑定调用方法 |
0xb6 | invokevirtual |
indexbyte1 indexbyte2 |
运行时方法绑定调用方法 |
0xb8 | invokestatic |
indexbyte1 indexbyte2 |
调用静态方法 |
0xb9 | invokeinterface |
indexbyte1 indexbyte2 count 0 |
调用接口方法 |
方法返回指令
指令码 | 操作码(助记符) | 描述(栈指操作数栈) |
---|---|---|
0xac | ireturn | 返回int类型值 |
0xad | lreturn | 返回long类型值 |
0xae | freturn | 返回float类型值 |
0xaf | dreturn | 返回double类型值 |
0xb0 | areturn | 返回引用类型值 |
0xb1 | return | void函数返回 |
线程同步指令
指令码 | 操作码(助记符) | 描述(栈指操作数栈) |
---|---|---|
0xc2 | monitorenter | 进入并获得对象监视器,获取对象锁,用于同步方法或者同步块 |
0xc3 | monitorexit | 释放并退出对象监视器,释放对象锁,用于同步方法或者同步块 |
wide指令
指令码 | 操作码(助记符) | 描述(栈指操作数栈) |
---|---|---|
0xc4 | wide | 使用附加字节扩展局部变量的宽度(iinc指令特殊) |
示例
方法的执行示例
Java源码:
public int inc() {int x;try {x = 1;return x;} catch (Exception e) {x = 2;return x;} finally {x = 3;}
}
编译后的ByteCode字节码及异常表:
public int inc();Code:Stack=1, Locals=5, Args_size=10: iconst_1 // try块中的x=11: istore_12: iload_1 // 保存x到returnValue中,此时x=13: istore 45: iconst_3 // finaly块中的x=36: istore_17: iload 4 // 将returnValue中的值放到栈顶,准备给ireturn返回9: ireturn10: astore_2 // 给catch中定义的Exception e赋值,存储在变量槽 2中11: iconst_2 // catch块中的x=212: istore_113: iload_1 // 保存x到returnValue中,此时x=214: istore 416: iconst_3 // finaly块中的x=317: istore_118: iload 4 // 将returnValue中的值放到栈顶,准备给ireturn返回20: ireturn21: astore_3 // 如果出现了不属于java.lang.Exception及其子类的异常才会走到这里22: iconst_3 // finaly块中的x=323: istore_124: aload_3 // 将异常放置到栈顶,并抛出25: athrowException table:from to target type0 5 10 Class java/lang/Exception0 5 21 any10 16 21 any
编译器为这段Java源码生成了三条异常表记录,对应三条可能出现的代码执行路径。从Java代码的语义上讲,这三条执行路径分别为:
- 如果try语句块中出现属于Exception或其子类的异常,转到catch语句块处理;
- 如果try语句块中出现不属于Exception或其子类的异常,转到finally语句块处理;
- 如果catch语句块中出现任何异常,转到finally语句块处理。
参考
《深入理解JAVA虚拟机》
https://www.cnblogs.com/honger/p/6815198.html
深入理解JVM - 虚拟机字节码指令集相关推荐
- 深入理解Java虚拟机 - 字节码指令集
Java虚拟机的指令由一个字节长度的. 代表着某种特定操作含义的数字(称为操作码,Opcode)以及跟随其后的零至多个代表此操作所需参数(称为操作数,Operands)而构成. 字节码与数据类型 在J ...
- 第六章JVM虚拟机字节码执行引擎——类文件和类加载之前必看
文章目录 虚拟机字节码执行引擎 运行时栈帧结构 局部变量表(Local Variables) 操作数栈 动态链接(Dynamic Linking) 方法返回地址 附加信息 方法调用 解析 分派 虚方法 ...
- 深入理解JVM一字节码执行
文章目录 前言 栈帧结构 每个方法调用开始到退出,都对应着一个"栈帧"进站与出站. 运行时栈帧 栈帧-局部变量表 栈帧-操作数栈(Operand Stack) 栈帧-动态连接 栈帧 ...
- 深入理解JVM 一字节码详解
今天继续总结JVM,计划本周完成这个系列的整理.总结. 本节内容枯燥,胆小者勿入! Write Once,Run Anywhere--byteCode byteCode 平台无关 java通过存储编译 ...
- [三] java虚拟机 JVM字节码 指令集 bytecode 操作码 指令分类用法 助记符
说明,本文的目的在于从宏观逻辑上介绍清楚绝大多数的字节码指令的含义以及分类 只要认真阅读本文必然能够对字节码指令集有所了解 如果需要了解清楚每一个指令的具体详尽用法,请参阅虚拟机规范 指令简介 计算机 ...
- 【深入理解JVM】JVM字节码指令集
Java 虚拟机的指令由一个字节长度的.代表着某种特定操作含义的操作码(Opcode)以及跟随其后的零至多个代表此操作所需参数的操作数(Operands)所构成.虚拟机中许多指令并不包含操作数,只有一 ...
- 【深入理解JVM 六】虚拟机字节码执行系统
前边几篇Blog分别介绍了JVM的类加载机制.运行时数据区域,那么字节码文件已然被加载到了JVM的内存中来,接下来就是执行它了.虚拟机字节码执行引擎的位置如下图所示,除了本地方法栈,其他运行时数据区域 ...
- JVM字节码指令集大全及其介绍
Java是怎么跨平台的 我们上计算机课的时候老师讲过:"计算机只能识别0和1,所以我们写的程序要经过编译器翻译成0和1组成的二进制格式计算机才能执行".我们编译后产生的.class ...
- jvm(8)-虚拟机字节码执行引擎
[0]README 0.1)本文转自 "深入理解jvm",旨在学习 虚拟机字节码执行引擎 的基础知识: [1]概述 1)物理机和虚拟机的执行引擎: 物理机的执行引擎是直接建立在处理 ...
- Java指令全集_Java的JVM字节码指令集详解
本文详细介绍了如何使用javap查看java方法中的字节码.以及各种字节码的含义,并且配以完善的案例,一步步,从头到尾带领大家翻译javap的输出.在文末还附有JVM字节码指令集表. 本文不适合没有J ...
最新文章
- ASP.NET 2.0站点登录、导航与权限管理
- java 组合框_Java知多少(88)列表和组合框
- Be a new gentleman
- MS-SQL中的事务
- flutter 弹幕插件_Flutter 实现虎牙/斗鱼 弹幕效果 | 秒速技术
- 硬件知识:串口通讯的起始、数据、停止位是怎么分配的?
- php bind_param,php – 在mysqli bind_param中绑定动态变量
- VS2005与紫光拼音输入法全角问题的解决办法
- 示例项目-京东商城图书分类排行榜抓取
- 1123581321递归算法java_斐波那契数列的算法
- Keras LSTM输入的input的特征和时间纬度
- Postman模拟接口API:接收请求报文equest,响应返回固定报文response
- 《精通python设计模式》读书笔记之——结构型设计模式
- Predictable Programming on a Precision Timed Architecture
- 闪亮玻璃图标悬浮效果
- 2021 谁怕?一蓑烟雨任平生
- 关于EFI的一点介绍
- 从tushare获取股票基金数据制作k线图
- python3d相册源代码_js和CSS3炫酷3D相册展示
- 小米电视还原设置的方法步骤