java - 怎么理解JVM中的iload和istore指令
問題描述
我最近在學(xué)習(xí)JVM,被istore和iload兩條指令困擾了。以下是我查看《Java虛擬機(jī)規(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)將計(jì)算結(jié)果壓入到操作數(shù)棧了,而指令6又是把結(jié)果壓入到操作數(shù)棧,這不是重復(fù)工作嗎。如果存入操作數(shù)棧的意義是為了可以store到局部變量表中,那第6步又為什么要load到操作數(shù)棧上。不知道,是不是我哪步理解錯了,謝謝指點(diǎn)。
問題解答
回答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:樓上正解,其實(shí)結(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=和; 此時(shí)棧為空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. javascript - 有適合開發(fā)手機(jī)端Html5網(wǎng)頁小游戲的前端框架嗎?2. javascript - arguments.callee3. javascript - js setTimeout在雙重for循環(huán)中如何使用?4. java - 線上應(yīng)用,如果數(shù)據(jù)庫操作失敗的話應(yīng)該如何處理?5. atom開始輸入!然后按tab只有空格出現(xiàn)沒有html格式出現(xiàn)6. node.js - 阿里云ECS,阿里云Docker,還有Leancloud的LeanEgine,哪個更適合NodeJs WebApp?7. java - 創(chuàng)建maven項(xiàng)目失敗了 求解決方法8. mac里的docker如何命令行開啟呢?9. mysql - 這種分級一對多,且分級不平衡的模型該怎么設(shè)計(jì)表?10. mysql - linux連接數(shù)據(jù)庫報(bào)錯
