java - 怎么理解JVM中的iload和istore指令
問題描述
我最近在學(xué)習(xí)JVM,被istore和iload兩條指令困擾了。以下是我查看《Java虛擬機規(guī)范》得到的解釋
將一個局部變量加載到操縱棧的指令包括:iload、iload_、lload…
將一個數(shù)值從操作數(shù)棧存儲到局部變量表的指令包括:istore、istore_、lstore…
下面是我的java代碼
public static int add(int a,int b){int c=0;c=a+b;return c; }
下面是編譯后的字節(jié)碼,也加上了我的理解,如果解釋不恰當(dāng),謝謝指出
0: iconst_0//常量0壓入操作數(shù)棧1: istore_2//彈出操作數(shù)棧棧頂元素,保存到局部變量表第2個位置2: iload_0 //第0個變量壓入操作數(shù)棧3: iload_1 //第1個變量壓入操作數(shù)棧4: iadd //操作數(shù)棧中的前兩個int相加,并將結(jié)果壓入操作數(shù)棧頂5: istore_2//彈出操作數(shù)棧棧頂元素,保存到局部變量表第2個位置6: iload_2 //加載局部變量表的第2個變量到操作數(shù)棧頂7: ireturn //返回
從上面字節(jié)碼的分析看,指令4已經(jīng)將計算結(jié)果壓入到操作數(shù)棧了,而指令6又是把結(jié)果壓入到操作數(shù)棧,這不是重復(fù)工作嗎。如果存入操作數(shù)棧的意義是為了可以store到局部變量表中,那第6步又為什么要load到操作數(shù)棧上。不知道,是不是我哪步理解錯了,謝謝指點。
問題解答
回答1:如果把代碼換成
public static int add(int a,int b){int c=0;return a+b; }
那么指令對應(yīng)就是:
0: iconst_0 1: istore_2 2: iload_0 3: iload_1 4: iadd 5: ireturn
編譯器就是按照代碼來生成的,如果直接 return a + b,那么也不會多出來第五步和第六步。
回答2:樓上正解,其實結(jié)合代碼看下就可以很明白的看出原因了。
首先這個方法是靜態(tài)方法,所以局部變量數(shù)組【0】【1】【2】對應(yīng)的變量分別為a、b、c;
0: iconst_0//常量0入棧1: istore_2//將棧頂出棧,即c=0;2: iload_0 //復(fù)制a變量的值入棧3: iload_1 //復(fù)制b變量的值入棧4: iadd //將棧頂兩個元素出棧,做加法,然后把結(jié)果再入棧(即a,b出棧,將a+b入棧)5: istore_2//棧頂元素出棧,即c=和; 此時棧為空6: iload_2 //將c賦值壓入棧7: ireturn //返回棧頂元素回答3:
一個小錯誤,局部變量表的index是從0開始的。
編譯器生成的字節(jié)碼完全是按照方法中的語義生成的,沒有太多優(yōu)化。
iadd指令對應(yīng)的a+b中加法操作,下一步的istore_2對應(yīng)的就是c=的賦值操作,也就是保存到局部變量表,后面的iload_2對應(yīng)的就是return中取c的值。
相關(guān)文章:
1. css - 求推薦幾款好用的移動端頁面布局調(diào)試工具呢?2. javascript - 百度echarts series數(shù)據(jù)更新問題3. css3 - css before 中文亂碼?4. php - 第三方支付平臺在很短時間內(nèi)多次異步通知,訂單多次確認(rèn)收款5. Mysql && Redis 并發(fā)問題6. javascript - node服務(wù)端渲染的困惑7. javascript - 請問一下組件的生命周期beforeDestory是在什么情況下面觸發(fā)的呢?8. mysql - 一個表和多個表是多對多的關(guān)系,該怎么設(shè)計9. python - type函數(shù)問題10. mysql新建字段時 timestamp NOT NULL DEFAULT ’0000-00-00 00:00:00’ 報錯
