《Java核心技術》 JVM指令集

Yes, it is as you guessed. The JVM/JRE uses Java bytecode as its instruction set and each JVM needs to be compiled on and be runnable on the native/local hardware (and therefore the local instruction set). This diagram from Wikipedia illustrates this well I think:

The JRE/JVM needs to be compiled for the specific hardware it runs on, though the Java bytecode definitions and interpretations by the JVM itself stay the same. As you point out, the Java bytecode can be seen as a kind of abstraction layer between the Java source code and the local machine/binary code. It does allow for a separation of concerns between the typical Java programmer and needing to know anything machine-specific, as almost all of that is handled by the JVM/JRE.

cafe babe 0000 0032 0022 0a00 0600 1409
0015 0016 0800 170a 0018 0019 0700 1a07
001b 0100 063c 696e 6974 3e01 0003 2829
5601 0004 436f 6465 0100 0f4c 696e 654e
756d 6265 7254 6162 6c65 0100 124c 6f63
616c 5661 7269 6162 6c65 5461 626c 6501
0004 7468 6973 0100 0c4c 4865 6c6c 6f57
6f72 6c64 3b01 0004 6d61 696e 0100 1628
5b4c 6a61 7661 2f6c 616e 672f 5374 7269
6e67 3b29 5601 0004 6172 6773 0100 135b
4c6a 6176 612f 6c61 6e67 2f53 7472 696e
673b 0100 0a53 6f75 7263 6546 696c 6501
000f 4865 6c6c 6f57 6f72 6c64 2e6a 6176
610c 0007 0008 0700 1c0c 001d 001e 0100
0b48 656c 6c6f 2c57 6f72 6c64 0700 1f0c
0020 0021 0100 0a48 656c 6c6f 576f 726c
6401 0010 6a61 7661 2f6c 616e 672f 4f62
6a65 6374 0100 106a 6176 612f 6c61 6e67
2f53 7973 7465 6d01 0003 6f75 7401 0015
4c6a 6176 612f 696f 2f50 7269 6e74 5374
7265 616d 3b01 0013 6a61 7661 2f69 6f2f
5072 696e 7453 7472 6561 6d01 0007 7072
696e 746c 6e01 0015 284c 6a61 7661 2f6c
616e 672f 5374 7269 6e67 3b29 5600 2100
0500 0600 0000 0000 0200 0100 0700 0800
0100 0900 0000 2f00 0100 0100 0000 052a
b700 01b1 0000 0002 000a 0000 0006 0001
0000 0007 000b 0000 000c 0001 0000 0005
000c 000d 0000 0009 000e 000f 0001 0009
0000 0037 0002 0001 0000 0009 b200 0212
03b6 0004 b100 0000 0200 0a00 0000 0a00
0200 0000 0900 0800 0a00 0b00 0000 0c00
0100 0000 0900 1000 1100 0000 0100 1200
0000 0200 13


$ javap -c HelloWorld.class 
Compiled from "HelloWorld.java"
public class HelloWorld {
  public HelloWorld();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  public static void main(java.lang.String[]);
    Code:
       0: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
       3: ldc           #3                  // String Hello,World
       5: invokevirtual #4                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
       8: return
}


指令碼 助記符 說明
0×00 nop 什么都不做
0×01 aconst_null 將null推送至棧頂
0×02 iconst_m1 將int型-1推送至棧頂
0×03 iconst_0 將int型0推送至棧頂

0×04
iconst_1
將int型1推送至棧頂

0×05
iconst_2
將int型2推送至棧頂

0×06
iconst_3
將int型3推送至棧頂

0×07
iconst_4
將int型4推送至棧頂

0×08
iconst_5
將int型5推送至棧頂

0×09
lconst_0
將long型0推送至棧頂

0x0a
lconst_1
將long型1推送至棧頂

0x0b
fconst_0
將float型0推送至棧頂

0x0c
fconst_1
將float型1推送至棧頂

0x0d
fconst_2
將float型2推送至棧頂

0x0e
dconst_0
將double型0推送至棧頂

0x0f
dconst_1
將double型1推送至棧頂

0×10
bipush
將單字節的常量值(-128~127)推送至棧頂

0×11
sipush
將一個短整型常量值(-32768~32767)推送至棧頂

0×12
ldc
將int, float或String型常量值從常量池中推送至棧頂

0×13
ldc_w
將int, float或String型常量值從常量池中推送至棧頂(寬索引)

0×14
ldc2_w
將long或double型常量值從常量池中推送至棧頂(寬索引)

0×15
iload
將指定的int型本地變量推送至棧頂

0×16
lload
將指定的long型本地變量推送至棧頂

0×17
fload
將指定的float型本地變量推送至棧頂

0×18
dload
將指定的double型本地變量推送至棧頂

0×19
aload
將指定的引用類型本地變量推送至棧頂

0x1a
iload_0
將第0個int型本地變量推送至棧頂

0x1b
iload_1
將第1個int型本地變量推送至棧頂

0x1c
iload_2
將第2個int型本地變量推送至棧頂

0x1d
iload_3
將第3個int型本地變量推送至棧頂

0x1e
lload_0
將第0個long型本地變量推送至棧頂

0x1f
lload_1
將第1個long型本地變量推送至棧頂

0×20
lload_2
將第2個long型本地變量推送至棧頂

0×21
lload_3
將第3個long型本地變量推送至棧頂

0×22
fload_0
將第0個float型本地變量推送至棧頂

0×23
fload_1
將第1個float型本地變量推送至棧頂

0×24
fload_2
將第2個float型本地變量推送至棧頂

0×25
fload_3
將第3個float型本地變量推送至棧頂

0×26
dload_0
將第0個double型本地變量推送至棧頂

0×27
dload_1
將第1個double型本地變量推送至棧頂

0×28
dload_2
將第2個double型本地變量推送至棧頂

0×29
dload_3
將第3個double型本地變量推送至棧頂

0x2a
aload_0
將第0個引用類型本地變量推送至棧頂

0x2b
aload_1
將第1個引用類型本地變量推送至棧頂

0x2c
aload_2
將第2個引用類型本地變量推送至棧頂

0x2d
aload_3
將第3個引用類型本地變量推送至棧頂

0x2e
iaload
將int型數組指定索引的值推送至棧頂

0x2f
laload
將long型數組指定索引的值推送至棧頂

0×30
faload
將float型數組指定索引的值推送至棧頂

0×31
daload
將double型數組指定索引的值推送至棧頂

0×32
aaload
將引用型數組指定索引的值推送至棧頂

0×33
baload
將boolean或byte型數組指定索引的值推送至棧頂

0×34
caload
將char型數組指定索引的值推送至棧頂

0×35
saload
將short型數組指定索引的值推送至棧頂

0×36
istore
將棧頂int型數值存入指定本地變量

0×37
lstore
將棧頂long型數值存入指定本地變量

0×38
fstore
將棧頂float型數值存入指定本地變量

0×39
dstore
將棧頂double型數值存入指定本地變量

0x3a
astore
將棧頂引用型數值存入指定本地變量

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個本地變量

0×40
lstore_1
將棧頂long型數值存入第1個本地變量

0×41
lstore_2
將棧頂long型數值存入第2個本地變量

0×42
lstore_3
將棧頂long型數值存入第3個本地變量

0×43
fstore_0
將棧頂float型數值存入第0個本地變量

0×44
fstore_1
將棧頂float型數值存入第1個本地變量

0×45
fstore_2
將棧頂float型數值存入第2個本地變量

0×46
fstore_3
將棧頂float型數值存入第3個本地變量

0×47
dstore_0
將棧頂double型數值存入第0個本地變量

0×48
dstore_1
將棧頂double型數值存入第1個本地變量

0×49
dstore_2
將棧頂double型數值存入第2個本地變量

0x4a
dstore_3
將棧頂double型數值存入第3個本地變量

0x4b
astore_0
將棧頂引用型數值存入第0個本地變量

0x4c
astore_1
將棧頂引用型數值存入第1個本地變量

0x4d
astore_2
將棧頂引用型數值存入第2個本地變量

0x4e
astore_3
將棧頂引用型數值存入第3個本地變量

0x4f
iastore
將棧頂int型數值存入指定數組的指定索引位置

0×50
lastore
將棧頂long型數值存入指定數組的指定索引位置

0×51
fastore
將棧頂float型數值存入指定數組的指定索引位置

0×52
dastore
將棧頂double型數值存入指定數組的指定索引位置

0×53
aastore
將棧頂引用型數值存入指定數組的指定索引位置

0×54
bastore
將棧頂boolean或byte型數值存入指定數組的指定索引位置

0×55
castore
將棧頂char型數值存入指定數組的指定索引位置

0×56
sastore
將棧頂short型數值存入指定數組的指定索引位置

0×57
pop
將棧頂數值彈出 (數值不能是long或double類型的)

0×58
pop2
將棧頂的一個(long或double類型的)或兩個數值彈出(其它)

0×59
dup
復制棧頂數值并將復制值壓入棧頂

0x5a
dup_x1
復制棧頂數值并將兩個復制值壓入棧頂

0x5b
dup_x2
復制棧頂數值并將三個(或兩個)復制值壓入棧頂

0x5c
dup2
復制棧頂一個(long或double類型的)或兩個(其它)數值并將復制值壓入棧頂

0x5d
dup2_x1
<待補充>

0x5e
dup2_x2
<待補充>

0x5f
swap
將棧最頂端的兩個數值互換(數值不能是long或double類型的)

0×60
iadd
將棧頂兩int型數值相加并將結果壓入棧頂

0×61
ladd
將棧頂兩long型數值相加并將結果壓入棧頂

0×62
fadd
將棧頂兩float型數值相加并將結果壓入棧頂

0×63
dadd
將棧頂兩double型數值相加并將結果壓入棧頂

0×64
isub
將棧頂兩int型數值相減并將結果壓入棧頂

0×65
lsub
將棧頂兩long型數值相減并將結果壓入棧頂

0×66
fsub
將棧頂兩float型數值相減并將結果壓入棧頂

0×67
dsub
將棧頂兩double型數值相減并將結果壓入棧頂

0×68
imul
將棧頂兩int型數值相乘并將結果壓入棧頂

0×69
lmul
將棧頂兩long型數值相乘并將結果壓入棧頂

0x6a
fmul
將棧頂兩float型數值相乘并將結果壓入棧頂

0x6b
dmul
將棧頂兩double型數值相乘并將結果壓入棧頂

0x6c
idiv
將棧頂兩int型數值相除并將結果壓入棧頂

0x6d
ldiv
將棧頂兩long型數值相除并將結果壓入棧頂

0x6e
fdiv
將棧頂兩float型數值相除并將結果壓入棧頂

0x6f
ddiv
將棧頂兩double型數值相除并將結果壓入棧頂

0×70
irem
將棧頂兩int型數值作取模運算并將結果壓入棧頂

0×71
lrem
將棧頂兩long型數值作取模運算并將結果壓入棧頂

0×72
frem
將棧頂兩float型數值作取模運算并將結果壓入棧頂

0×73
drem
將棧頂兩double型數值作取模運算并將結果壓入棧頂

0×74
ineg
將棧頂int型數值取負并將結果壓入棧頂

0×75
lneg
將棧頂long型數值取負并將結果壓入棧頂

0×76
fneg
將棧頂float型數值取負并將結果壓入棧頂

0×77
dneg
將棧頂double型數值取負并將結果壓入棧頂

0×78
ishl
將int型數值左移位指定位數并將結果壓入棧頂

0×79
lshl
將long型數值左移位指定位數并將結果壓入棧頂

0x7a
ishr
將int型數值右(符號)移位指定位數并將結果壓入棧頂

0x7b
lshr
將long型數值右(符號)移位指定位數并將結果壓入棧頂

0x7c
iushr
將int型數值右(無符號)移位指定位數并將結果壓入棧頂

0x7d
lushr
將long型數值右(無符號)移位指定位數并將結果壓入棧頂

0x7e
iand
將棧頂兩int型數值作“按位與”并將結果壓入棧頂

0x7f
land
將棧頂兩long型數值作“按位與”并將結果壓入棧頂

0×80
ior
將棧頂兩int型數值作“按位或”并將結果壓入棧頂

0×81
lor
將棧頂兩long型數值作“按位或”并將結果壓入棧頂

0×82
ixor
將棧頂兩int型數值作“按位異或”并將結果壓入棧頂

0×83
lxor
將棧頂兩long型數值作“按位異或”并將結果壓入棧頂

0×84
iinc
將指定int型變量增加指定值,可以有兩個變量,分別表示index, const,index指第index個int型本地變量,const增加的值

0×85
i2l
將棧頂int型數值強制轉換成long型數值并將結果壓入棧頂

0×86
i2f
將棧頂int型數值強制轉換成float型數值并將結果壓入棧頂

0×87
i2d
將棧頂int型數值強制轉換成double型數值并將結果壓入棧頂

0×88
l2i
將棧頂long型數值強制轉換成int型數值并將結果壓入棧頂

0×89
l2f
將棧頂long型數值強制轉換成float型數值并將結果壓入棧頂

0x8a
l2d
將棧頂long型數值強制轉換成double型數值并將結果壓入棧頂

0x8b
f2i
將棧頂float型數值強制轉換成int型數值并將結果壓入棧頂

0x8c
f2l
將棧頂float型數值強制轉換成long型數值并將結果壓入棧頂

0x8d
f2d
將棧頂float型數值強制轉換成double型數值并將結果壓入棧頂

0x8e
d2i
將棧頂double型數值強制轉換成int型數值并將結果壓入棧頂

0x8f
d2l
將棧頂double型數值強制轉換成long型數值并將結果壓入棧頂

0×90
d2f
將棧頂double型數值強制轉換成float型數值并將結果壓入棧頂

0×91
i2b
將棧頂int型數值強制轉換成byte型數值并將結果壓入棧頂

0×92
i2c
將棧頂int型數值強制轉換成char型數值并將結果壓入棧頂

0×93
i2s
將棧頂int型數值強制轉換成short型數值并將結果壓入棧頂

0×94
lcmp
比較棧頂兩long型數值大小,并將結果(1,0,-1)壓入棧頂

0×95
fcmpl
比較棧頂兩float型數值大小,并將結果(1,0,-1)壓入棧頂;當其中一個數值為NaN時,將-1壓入棧頂

0×96
fcmpg
比較棧頂兩float型數值大小,并將結果(1,0,-1)壓入棧頂;當其中一個數值為NaN時,將1壓入棧頂

0×97
dcmpl
比較棧頂兩double型數值大小,并將結果(1,0,-1)壓入棧頂;當其中一個數值為NaN時,將-1壓入棧頂

0×98
dcmpg
比較棧頂兩double型數值大小,并將結果(1,0,-1)壓入棧頂;當其中一個數值為NaN時,將1壓入棧頂

0×99
ifeq
當棧頂int型數值等于0時跳轉

0x9a
ifne
當棧頂int型數值不等于0時跳轉

0x9b
iflt
當棧頂int型數值小于0時跳轉

0x9c
ifge
當棧頂int型數值大于等于0時跳轉

0x9d
ifgt
當棧頂int型數值大于0時跳轉

0x9e
ifle
當棧頂int型數值小于等于0時跳轉

0x9f
if_icmpeq
比較棧頂兩int型數值大小,當結果等于0時跳轉

0xa0
if_icmpne
比較棧頂兩int型數值大小,當結果不等于0時跳轉

0xa1
if_icmplt
比較棧頂兩int型數值大小,當結果小于0時跳轉

0xa2
if_icmpge
比較棧頂兩int型數值大小,當結果大于等于0時跳轉

0xa3
if_icmpgt
比較棧頂兩int型數值大小,當結果大于0時跳轉

0xa4
if_icmple
比較棧頂兩int型數值大小,當結果小于等于0時跳轉

0xa5
if_acmpeq
比較棧頂兩引用型數值,當結果相等時跳轉

0xa6
if_acmpne
比較棧頂兩引用型數值,當結果不相等時跳轉

0xa7
goto
無條件跳轉

0xa8
jsr
跳轉至指定16位offset位置,并將jsr下一條指令地址壓入棧頂

0xa9
ret
返回至本地變量指定的index的指令位置(一般與jsr, jsr_w聯合使用)

0xaa
tableswitch
用于switch條件跳轉,case值連續(可變長度指令)

0xab
lookupswitch
用于switch條件跳轉,case值不連續(可變長度指令)

0xac
ireturn
從當前方法返回int

0xad
lreturn
從當前方法返回long

0xae
freturn
從當前方法返回float

0xaf
dreturn
從當前方法返回double

0xb0
areturn
從當前方法返回對象引用

0xb1
return
從當前方法返回void

0xb2
getstatic
獲取指定類的靜態域,并將其值壓入棧頂

0xb3
putstatic
為指定的類的靜態域賦值

0xb4
getfield
獲取指定類的實例域,并將其值壓入棧頂

0xb5
putfield
為指定的類的實例域賦值

0xb6
invokevirtual
調用實例方法

0xb7
invokespecial
調用超類構造方法,實例初始化方法,私有方法

0xb8
invokestatic
調用靜態方法

0xb9
invokeinterface
調用接口方法

0xba

0xbb
new
創建一個對象,并將其引用值壓入棧頂

0xbc
newarray
創建一個指定原始類型(如int, float, char…)的數組,并將其引用值壓入棧頂

0xbd
anewarray
創建一個引用型(如類,接口,數組)的數組,并將其引用值壓入棧頂

0xbe
arraylength
獲得數組的長度值并壓入棧頂

0xbf
athrow
將棧頂的異常拋出

0xc0
checkcast
檢驗類型轉換,檢驗未通過將拋出ClassCastException

0xc1
instanceof
檢驗對象是否是指定的類的實例,如果是將1壓入棧頂,否則將0壓入棧頂

0xc2
monitorenter
獲得對象的鎖,用于同步方法或同步塊

0xc3
monitorexit
釋放對象的鎖,用于同步方法或同步塊

0xc4
wide
當本地變量的索引超過255時使用該指令擴展索引寬度。

0xc5
multianewarray
create a new array of dimensions dimensions with elements of type identified by class reference in constant pool index (indexbyte1 << 8 + indexbyte2); the sizes of each dimension is identified by count1, [count2, etc.]

0xc6
ifnull
if value is null, branch to instruction at branchoffset (signed short constructed from unsigned bytes branchbyte1 << 8 + branchbyte2)

0xc7
ifnonnull
if value is not null, branch to instruction at branchoffset (signed short constructed from unsigned bytes branchbyte1 << 8 + branchbyte2)

0xc8
goto_w
goes to another instruction at branchoffset (signed int constructed from unsigned bytes branchbyte1 << 24 + branchbyte2 << 16 + branchbyte3 << 8 + branchbyte4)

0xc9
jsr_w
jump to subroutine at branchoffset (signed int constructed from unsigned bytes branchbyte1 << 24 + branchbyte2 << 16 + branchbyte3 << 8 + branchbyte4) and place the return address on the stack

0xca
breakpoint
reserved for breakpoints in Java debuggers; should not appear in any class file

0xcb-0xfd
未命名
these values are currently unassigned for opcodes and are reserved for future use

0xfe
impdep1
reserved for implementation-dependent operations within debuggers; should not appear in any class file

0xff
impdep2
reserved for implementation-dependent operations within debuggers; should not appear in any class file

有了以上指令集表,那么在查看字節碼就方便多了。
對應英文 => https://en.wikipedia.org/wiki/Java_bytecode_instruction_listings

來幾個sample:

public void sample1(){
int num = 5;
}

javap -c 查看字節碼

public void sample1();
Code:
0: iconst_5
1: istore_1
2: return

解釋

iconst_5 //將int型5推送至棧頂
istore_1 //將棧頂int型數值存入第1個本地變量
return //從當前方法返回void

public int sample2(int a, int b) {
return a + b;
}

字節碼及解釋

public int sample2(int, int);
Code:
0: iload_1 //將第1個int型本地變量推送至棧頂
1: iload_2 //將第2個int型本地變量推送至棧頂
2: iadd //將棧頂兩int型數值相加并將結果壓入棧頂
3: ireturn //從當前方法返回int

稍稍復雜點

public float sample3() {
float num = 0;
for (int i = 0; i < 5; i++) {
num *= i;
}
return num;
}

字節碼及解釋

public float sample3();
Code:
0: fconst_0 //將float型0推送至棧頂
1: fstore_1 //將棧頂float型數值存入第1個本地變量
2: iconst_0 //將int型0推送至棧頂,也就是for循環中的i = 0
3: istore_2 //將棧頂int型數值存入第2個本地變量
4: iload_2 //將第2個int型本地變量推送至棧頂
5: iconst_5 //將int型5推送至棧頂,也就是for循環中的 最大值5
6: if_icmpge 20 //比較棧頂兩int型數值大小,當結果大于等于0時跳轉,
//也就是比較0是否大于等于5,(cpmge指compare larger equals),如果是跳轉到到第20條指令
9: fload_1 //將第1個float型本地變量推送至棧頂,也就是變量num
10: iload_2 //將第2個int型本地變量推送至棧頂,也就是for循環中的變量i
11: i2f //int型強轉為float型,也就是把變量i強轉成float
12: fmul //將棧頂兩float型數值相乘并將結果壓入棧頂,也就是i與num相乘
13: fstore_1 //將棧頂float型數值存入第1個本地變量,也就是之前i與num的乘積
14: iinc 2, 1 //將指定int型變量增加指定值,將第2個int型本地變量增加1,
//可以看到,第2個int型本地變量就是之前的變量i
17: goto 4 //無條件跳轉到指令4,實現循環效果
20: fload_1 //將第1個float型本地變量推送至棧頂
21: freturn //從當前方法返回float

文章參考:

Java_bytecode_instruction_listings:

https://en.wikipedia.org/wiki/Java_bytecode_instruction_listings

JVM Opcode Reference:
http://homepages.inf.ed.ac.uk/kwxm/JVM/

jvm_instruction_set:

http://www.mobilefish.com/tutorials/java/java_quickguide_jvm_instruction_set.html

比如說,在多線程編程中,獲得一個對象的鎖,我們使用jvm指令集(jvm instruction set)中的monitorenter, monitorexit指令

Description: lock object for synchronized execution
Stack

Before  After
object to be locked       
Example

Jasm-------------------
aload_1      ; push object in local var 1
monitorenter      ; acquire lock on object
( do stuff to object )      ;
monitorexit      ; release lock
     ;
Exceptions

NullPointerException if object is null

問題1

那monitorenter, monitorexit指令最后經過jvm執行引擎,變成對應cpu機器的指令是什么?

問題2

最后編譯成匯編01字節碼,在OS&硬件層次上又是怎樣實現鎖對象的呢?

指令碼 助記符    說明
0x00 nop        無操作
0x01 aconst_null 將null推送至棧頂
0x02 iconst_m1    將int型-1推送至棧頂
0x03 iconst_0    將int型0推送至棧頂
0x04 iconst_1    將int型1推送至棧頂
0x05 iconst_2    將int型2推送至棧頂
0x06 iconst_3    將int型3推送至棧頂
0x07 iconst_4    將int型4推送至棧頂
0x08 iconst_5    將int型5推送至棧頂
0x09 lconst_0    將long型0推送至棧頂
0x0a lconst_1    將long型1推送至棧頂
0x0b fconst_0    將float型0推送至棧頂
0x0c fconst_1    將float型1推送至棧頂
0x0d fconst_2    將float型2推送至棧頂
0x0e dconst_0    將double型0推送至棧頂
0x0f dconst_1    將double型1推送至棧頂
0x10 bipush    將單字節的常量值(-128~127)推送至棧頂
0x11 sipush    將一個短整型常量值(-32768~32767)推送至棧頂
0x12 ldc    將int, float或String型常量值從常量池中推送至棧頂
0x13 ldc_w    將int, float或String型常量值從常量池中推送至棧頂(寬索引)
0x14 ldc2_w    將long或double型常量值從常量池中推送至棧頂(寬索引)
0x15 iload    將指定的int型本地變量推送至棧頂
0x16 lload    將指定的long型本地變量推送至棧頂
0x17 fload    將指定的float型本地變量推送至棧頂
0x18 dload    將指定的double型本地變量推送至棧頂
0x19 aload    將指定的引用類型本地變量推送至棧頂
0x1a iload_0    將第一個int型本地變量推送至棧頂
0x1b iload_1    將第二個int型本地變量推送至棧頂
0x1c iload_2    將第三個int型本地變量推送至棧頂
0x1d iload_3    將第四個int型本地變量推送至棧頂
0x1e lload_0    將第一個long型本地變量推送至棧頂
0x1f lload_1    將第二個long型本地變量推送至棧頂
0x20 lload_2    將第三個long型本地變量推送至棧頂
0x21 lload_3    將第四個long型本地變量推送至棧頂
0x22 fload_0    將第一個float型本地變量推送至棧頂
0x23 fload_1    將第二個float型本地變量推送至棧頂
0x24 fload_2    將第三個float型本地變量推送至棧頂
0x25 fload_3    將第四個float型本地變量推送至棧頂
0x26 dload_0    將第一個double型本地變量推送至棧頂
0x27 dload_1    將第二個double型本地變量推送至棧頂
0x28 dload_2    將第三個double型本地變量推送至棧頂
0x29 dload_3    將第四個double型本地變量推送至棧頂
0x2a aload_0    將第一個引用類型本地變量推送至棧頂
0x2b aload_1    將第二個引用類型本地變量推送至棧頂
0x2c aload_2    將第三個引用類型本地變量推送至棧頂
0x2d aload_3    將第四個引用類型本地變量推送至棧頂
0x2e iaload    將int型數組指定索引的值推送至棧頂
0x2f laload    將long型數組指定索引的值推送至棧頂
0x30 faload    將float型數組指定索引的值推送至棧頂
0x31 daload    將double型數組指定索引的值推送至棧頂
0x32 aaload    將引用型數組指定索引的值推送至棧頂
0x33 baload    將boolean或byte型數組指定索引的值推送至棧頂
0x34 caload    將char型數組指定索引的值推送至棧頂
0x35 saload    將short型數組指定索引的值推送至棧頂
0x36 istore    將棧頂int型數值存入指定本地變量
0x37 lstore    將棧頂long型數值存入指定本地變量
0x38 fstore    將棧頂float型數值存入指定本地變量
0x39 dstore    將棧頂double型數值存入指定本地變量
0x3a astore    將棧頂引用型數值存入指定本地變量
0x3b istore_0    將棧頂int型數值存入第一個本地變量
0x3c istore_1    將棧頂int型數值存入第二個本地變量
0x3d istore_2    將棧頂int型數值存入第三個本地變量
0x3e istore_3    將棧頂int型數值存入第四個本地變量
0x3f lstore_0    將棧頂long型數值存入第一個本地變量
0x40 lstore_1    將棧頂long型數值存入第二個本地變量
0x41 lstore_2    將棧頂long型數值存入第三個本地變量
0x42 lstore_3    將棧頂long型數值存入第四個本地變量
0x43 fstore_0    將棧頂float型數值存入第一個本地變量
0x44 fstore_1    將棧頂float型數值存入第二個本地變量
0x45 fstore_2    將棧頂float型數值存入第三個本地變量
0x46 fstore_3    將棧頂float型數值存入第四個本地變量
0x47 dstore_0    將棧頂double型數值存入第一個本地變量
0x48 dstore_1    將棧頂double型數值存入第二個本地變量
0x49 dstore_2    將棧頂double型數值存入第三個本地變量
0x4a dstore_3    將棧頂double型數值存入第四個本地變量
0x4b astore_0    將棧頂引用型數值存入第一個本地變量
0x4c astore_1    將棧頂引用型數值存入第二個本地變量
0x4d astore_2    將棧頂引用型數值存入第三個本地變量
0x4e astore_3    將棧頂引用型數值存入第四個本地變量
0x4f iastore    將棧頂int型數值存入指定數組的指定索引位置
0x50 lastore    將棧頂long型數值存入指定數組的指定索引位置
0x51 fastore    將棧頂float型數值存入指定數組的指定索引位置
0x52 dastore    將棧頂double型數值存入指定數組的指定索引位置
0x53 aastore    將棧頂引用型數值存入指定數組的指定索引位置
0x54 bastore    將棧頂boolean或byte型數值存入指定數組的指定索引位置
0x55 castore    將棧頂char型數值存入指定數組的指定索引位置
0x56 sastore    將棧頂short型數值存入指定數組的指定索引位置
0x57 pop     將棧頂數值彈出 (數值不能是long或double類型的)
0x58 pop2    將棧頂的一個(long或double類型的)或兩個數值彈出(其它)
0x59 dup     復制棧頂數值并將復制值壓入棧頂
0x5a dup_x1    復制棧頂數值并將兩個復制值壓入棧頂
0x5b dup_x2    復制棧頂數值并將三個(或兩個)復制值壓入棧頂
0x5c dup2    復制棧頂一個(long或double類型的)或兩個(其它)數值并將復制值壓入棧頂
0x5d dup2_x1    <待補充>
0x5e dup2_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型數值取負并將結果壓入棧頂
0x78 ishl    將int型數值左移位指定位數并將結果壓入棧頂
0x79 lshl    將long型數值左移位指定位數并將結果壓入棧頂
0x7a ishr    將int型數值右(符號)移位指定位數并將結果壓入棧頂
0x7b lshr    將long型數值右(符號)移位指定位數并將結果壓入棧頂
0x7c iushr    將int型數值右(無符號)移位指定位數并將結果壓入棧頂
0x7d lushr    將long型數值右(無符號)移位指定位數并將結果壓入棧頂
0x7e iand    將棧頂兩int型數值作“按位與”并將結果壓入棧頂
0x7f land    將棧頂兩long型數值作“按位與”并將結果壓入棧頂
0x80 ior     將棧頂兩int型數值作“按位或”并將結果壓入棧頂
0x81 lor     將棧頂兩long型數值作“按位或”并將結果壓入棧頂
0x82 ixor    將棧頂兩int型數值作“按位異或”并將結果壓入棧頂
0x83 lxor    將棧頂兩long型數值作“按位異或”并將結果壓入棧頂
0x84 iinc    將指定int型變量增加指定值(i++, i--, i+=2)
0x85 i2l     將棧頂int型數值強制轉換成long型數值并將結果壓入棧頂
0x86 i2f     將棧頂int型數值強制轉換成float型數值并將結果壓入棧頂
0x87 i2d     將棧頂int型數值強制轉換成double型數值并將結果壓入棧頂
0x88 l2i     將棧頂long型數值強制轉換成int型數值并將結果壓入棧頂
0x89 l2f     將棧頂long型數值強制轉換成float型數值并將結果壓入棧頂
0x8a l2d     將棧頂long型數值強制轉換成double型數值并將結果壓入棧頂
0x8b f2i     將棧頂float型數值強制轉換成int型數值并將結果壓入棧頂
0x8c f2l     將棧頂float型數值強制轉換成long型數值并將結果壓入棧頂
0x8d f2d     將棧頂float型數值強制轉換成double型數值并將結果壓入棧頂
0x8e d2i     將棧頂double型數值強制轉換成int型數值并將結果壓入棧頂
0x8f d2l     將棧頂double型數值強制轉換成long型數值并將結果壓入棧頂
0x90 d2f     將棧頂double型數值強制轉換成float型數值并將結果壓入棧頂
0x91 i2b     將棧頂int型數值強制轉換成byte型數值并將結果壓入棧頂
0x92 i2c     將棧頂int型數值強制轉換成char型數值并將結果壓入棧頂
0x93 i2s     將棧頂int型數值強制轉換成short型數值并將結果壓入棧頂
0x94 lcmp    比較棧頂兩long型數值大小,并將結果(1,0,-1)壓入棧頂
0x95 fcmpl    比較棧頂兩float型數值大小,并將結果(1,0,-1)壓入棧頂;當其中一個數值為NaN時,將-1壓入棧頂
0x96 fcmpg    比較棧頂兩float型數值大小,并將結果(1,0,-1)壓入棧頂;當其中一個數值為NaN時,將1壓入棧頂
0x97 dcmpl    比較棧頂兩double型數值大小,并將結果(1,0,-1)壓入棧頂;當其中一個數值為NaN時,將-1壓入棧頂
0x98 dcmpg    比較棧頂兩double型數值大小,并將結果(1,0,-1)壓入棧頂;當其中一個數值為NaN時,將1壓入棧頂
0x99 ifeq    當棧頂int型數值等于0時跳轉
0x9a ifne    當棧頂int型數值不等于0時跳轉
0x9b iflt    當棧頂int型數值小于0時跳轉
0x9c ifge    當棧頂int型數值大于等于0時跳轉
0x9d ifgt    當棧頂int型數值大于0時跳轉
0x9e ifle    當棧頂int型數值小于等于0時跳轉
0x9f if_icmpeq    比較棧頂兩int型數值大小,當結果等于0時跳轉
0xa0 if_icmpne    比較棧頂兩int型數值大小,當結果不等于0時跳轉
0xa1 if_icmplt    比較棧頂兩int型數值大小,當結果小于0時跳轉
0xa2 if_icmpge    比較棧頂兩int型數值大小,當結果大于等于0時跳轉
0xa3 if_icmpgt    比較棧頂兩int型數值大小,當結果大于0時跳轉
0xa4 if_icmple    比較棧頂兩int型數值大小,當結果小于等于0時跳轉
0xa5 if_acmpeq    比較棧頂兩引用型數值,當結果相等時跳轉
0xa6 if_acmpne    比較棧頂兩引用型數值,當結果不相等時跳轉
0xa7 goto    無條件跳轉
0xa8 jsr     跳轉至指定16位offset位置,并將jsr下一條指令地址壓入棧頂
0xa9 ret     返回至本地變量指定的index的指令位置(一般與jsr, jsr_w聯合使用)
0xaa tableswitch    用于switch條件跳轉,case值連續(可變長度指令)
0xab lookupswitch    用于switch條件跳轉,case值不連續(可變長度指令)
0xac ireturn    從當前方法返回int
0xad lreturn    從當前方法返回long
0xae freturn    從當前方法返回float
0xaf dreturn    從當前方法返回double
0xb0 areturn    從當前方法返回對象引用
0xb1 return    從當前方法返回void
0xb2 getstatic    獲取指定類的靜態域,并將其值壓入棧頂
0xb3 putstatic    為指定的類的靜態域賦值
0xb4 getfield    獲取指定類的實例域,并將其值壓入棧頂
0xb5 putfield    為指定的類的實例域賦值
0xb6 invokevirtual    調用實例方法
0xb7 invokespecial    調用超類構造方法,實例初始化方法,私有方法
0xb8 invokestatic    調用靜態方法
0xb9 invokeinterface 調用接口方法
0xba --
0xbb new     創建一個對象,并將其引用值壓入棧頂
0xbc newarray    創建一個指定原始類型(如int, float, char…)的數組,并將其引用值壓入棧頂
0xbd anewarray    創建一個引用型(如類,接口,數組)的數組,并將其引用值壓入棧頂
0xbe arraylength 獲得數組的長度值并壓入棧頂
0xbf athrow    將棧頂的異常拋出
0xc0 checkcast    檢驗類型轉換,檢驗未通過將拋出ClassCastException
0xc1 instanceof 檢驗對象是否是指定的類的實例,如果是將1壓入棧頂,否則將0壓入棧頂
0xc2 monitorenter    獲得對象的鎖,用于同步方法或同步塊
0xc3 monitorexit    釋放對象的鎖,用于同步方法或同步塊
0xc4 wide    <待補充>
0xc5 multianewarray 創建指定類型和指定維度的多維數組(執行該指令時,操作棧中必須包含各維度的長度值),并將其引用值壓入棧頂
0xc6 ifnull    為null時跳轉
0xc7 ifnonnull    不為null時跳轉
0xc8 goto_w    無條件跳轉
0xc9 jsr_w    跳轉至指定32位offset位置,并將jsr_w下一條指令地址壓入棧頂
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,283評論 6 530
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 97,947評論 3 413
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 175,094評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,485評論 1 308
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,268評論 6 405
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,817評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,906評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,039評論 0 285
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,551評論 1 331
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,502評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,662評論 1 366
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,188評論 5 356
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 43,907評論 3 345
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,304評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,563評論 1 281
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,255評論 3 389
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,637評論 2 370

推薦閱讀更多精彩內容