本文繼續(xù)跟隨書上所學進行實踐妨猩。
Jvm是基于棧架構(gòu)潜叛、Dalvik是基于寄存器架構(gòu)。
還是寫一段簡單的代碼計算(a + b) * (a - b)來理解下⊥担看看二者的不同的之處销斟。
public class Hello{
public int foo(int a, int b){
return (a + b) * (a - b);
}
public static void main(String[] argc){
Hello hello = new Hello();
System.out.println(hello.foo(5, 3));
}
}
JVM
先來看Jvm:javac先編譯一下,生成java字節(jié)碼椒舵。然后用javap -c對代碼進行反匯編
$ javap -c /Users/johnhao/Downloads/Train/Hello.class
Compiled from "Hello.java"
public class Hello {
public Hello();
Code:
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: return
public int foo(int, int);
Code:
0: iload_1
1: iload_2
2: iadd
3: iload_1
4: iload_2
5: isub
6: imul
7: ireturn
可以看到foo函數(shù)一共占用了8個字節(jié)蚂踊,索引值從0開始,一直到7逮栅。iload_1其實是兩部分悴势,表示指令q前綴,表示操作類型是整數(shù)int型措伐;load表示將局部變量壓入棧中特纤。
過程也很清晰,iload_1先將參數(shù)1壓入棧侥加,iload_2再將參數(shù)2壓入棧捧存,iadd從棧頂彈出兩個int類型值,將值相加担败,然后把結(jié)果壓回棧頂昔穴。下面的第四條和第五條同樣分別將參數(shù)1和參數(shù)2分別壓入棧頂,isub從棧頂彈出兩個int類型值提前,將值相減吗货,然后把結(jié)果壓回棧頂。到這里棧頂就有兩個計算好的結(jié)果的int類型值了狈网,分別是(參數(shù)1 + 參數(shù)2)的結(jié)果和(參數(shù)1 - 參數(shù)2)的結(jié)果宙搬。第七條指令imul,從棧頂彈出兩個int類型值拓哺,將值相乘勇垛,然后把結(jié)果壓回棧頂。最后一條ireturn返回一個int類型值士鸥,到此foo函數(shù)就執(zhí)行完了闲孤。
Dalvik
先用dx命令生成Dalvik字節(jié)碼。因為我用的java8編譯的烤礁,所以需要更高版本的buildtools才能編譯成功讼积。
$ sh /Users/johnhao/Documents/android-sdk-macosx/build-tools/26.0.0/dx --dex --output=Hello.dex Hello.class
接下來使用dexdump -d來反編譯代碼
$ /Users/johnhao/Documents/android-sdk-macosx/build-tools/26.0.0/dexdump -d /Users/johnhao/Downloads/Train/Hello.dex
Processing '/Users/johnhao/Downloads/Train/Hello.dex'...
Opened '/Users/johnhao/Downloads/Train/Hello.dex', DEX version '035'
。鸽凶。币砂。省略一些其他的。玻侥。。
Virtual methods -
#0 : (in LHello;)
name : 'foo'
type : '(II)I'
access : 0x0001 (PUBLIC)
code -
registers : 5
ins : 3
outs : 0
insns size : 6 16-bit code units
000198: |[000198] Hello.foo:(II)I
0001a8: 9000 0304 |0000: add-int v0, v3, v4
0001ac: 9101 0304 |0002: sub-int v1, v3, v4
0001b0: b210 |0004: mul-int/2addr v0, v1
0001b2: 0f00 |0005: return v0
相對于Jvm亿蒸,Dalvik的字節(jié)碼要簡潔一些凑兰,只有4條就完成了以上操作掌桩。第一條add-int,將v3和v4寄存器的值相加姑食,然后保存結(jié)果到v0寄存器波岛。這里的v3和v4就是foo函數(shù)的第一個和第二個參數(shù)。第二條sub-int音半,將v3和v4寄存器的值相減则拷,然后把結(jié)果保存到v1寄存器中。第三條指令mul-int/2addr曹鸠,將v0和v1寄存器的值相乘煌茬,然后把結(jié)果存到v0寄存器。第四條指令返回v0寄存器的值彻桃。
詳細的v0坛善、v1這些參數(shù)的介紹:
了解Dalvik寄存器的v和p命名方式
借由這個簡單代碼,可以淺顯體會出基于棧和基于寄存器的不同之處邻眷。