美文网首页android基础知识
体会Java虚拟机和Dalvik虚拟机的架构不同之处

体会Java虚拟机和Dalvik虚拟机的架构不同之处

作者: h080294 | 来源:发表于2017-10-22 14:31 被阅读46次

    本文继续跟随书上所学进行实践。
    Jvm是基于栈架构、Dalvik是基于寄存器架构。
    还是写一段简单的代码计算(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字节码。然后用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函数一共占用了8个字节,索引值从0开始,一直到7。iload_1其实是两部分,表示指令q前缀,表示操作类型是整数int型;load表示将局部变量压入栈中。
    过程也很清晰,iload_1先将参数1压入栈,iload_2再将参数2压入栈,iadd从栈顶弹出两个int类型值,将值相加,然后把结果压回栈顶。下面的第四条和第五条同样分别将参数1和参数2分别压入栈顶,isub从栈顶弹出两个int类型值,将值相减,然后把结果压回栈顶。到这里栈顶就有两个计算好的结果的int类型值了,分别是(参数1 + 参数2)的结果和(参数1 - 参数2)的结果。第七条指令imul,从栈顶弹出两个int类型值,将值相乘,然后把结果压回栈顶。最后一条ireturn返回一个int类型值,到此foo函数就执行完了。

    Dalvik

    先用dx命令生成Dalvik字节码。因为我用的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的字节码要简洁一些,只有4条就完成了以上操作。第一条add-int,将v3和v4寄存器的值相加,然后保存结果到v0寄存器。这里的v3和v4就是foo函数的第一个和第二个参数。第二条sub-int,将v3和v4寄存器的值相减,然后把结果保存到v1寄存器中。第三条指令mul-int/2addr,将v0和v1寄存器的值相乘,然后把结果存到v0寄存器。第四条指令返回v0寄存器的值。

    详细的v0、v1这些参数的介绍:
    了解Dalvik寄存器的v和p命名方式

    借由这个简单代码,可以浅显体会出基于栈和基于寄存器的不同之处。

    相关文章

      网友评论

        本文标题:体会Java虚拟机和Dalvik虚拟机的架构不同之处

        本文链接:https://www.haomeiwen.com/subject/czzvuxtx.html