摘要:從字節(jié)碼的分析可以觀察到一個(gè)有趣的現(xiàn)象,再次看看我們的語句。這張表里每行的后面的數(shù)字代表源代碼的序號,冒號后面的數(shù)字代表字節(jié)碼里每行指令的序號。維護(hù)了源代碼同字節(jié)指令的映射關(guān)系,確保了代碼調(diào)試的順利進(jìn)行。
javap是JDK自帶的工具:
這篇文章使用下面這段簡單的Java代碼作為例子進(jìn)行講解。
class Outer { Nested nested; Nested getNested() { return nested; } } class Nested { Inner inner; Inner getInner() { return inner; } } class Inner { String foo; String getFoo() { return foo; } } public class NullableTest { public static Outer getInitializedOuter(){ Outer outer = new Outer(); outer.nested = new Nested(); outer.nested.inner = new Inner(); outer.nested.inner.foo = "Jerry"; return outer; } /* null pointer exception private static void way0(){ Outer outer = new Outer(); System.out.println(outer.nested.inner.foo); }*/ public static void way1(){ Outer outer = getInitializedOuter(); if (outer != null && outer.nested != null && outer.nested.inner != null) { System.out.println(outer.nested.inner.foo); } } public static void main(String[] args) { //way0(); way1(); } }
使用下面的命令行對NullableTest進(jìn)行反編譯,以java編譯器生成的字節(jié)碼:
javap -v NullableTest >c:code1.txt
查看方法way1()對應(yīng)的字節(jié)碼:
下面這個(gè)wiki包含了java字節(jié)碼里每個(gè)指令的具體說明:
https://en.wikipedia.org/wiki...
下面對NullableTest反編譯得到的字節(jié)碼做一些說明:
0: invokestatic #42 // Method getInitializedOuter:()Ljava8/Outer;
代表靜態(tài)方法getInitializedOuter的調(diào)用, Ljava8/Outer意思是該方法的返回類型是Outer
3: astore_0
將上述靜態(tài)方法調(diào)用返回的outer引用存儲到局部變量中,局部變量的id為0.
4: aload_0
因?yàn)樵谖仪懊娴腏ava源代碼中,我將靜態(tài)方法返回的對象引用同null做了比較,因此使用指令aload_0將存儲在代號為0的局部變量中的對象引用重新加載到棧上,此后才能和null做比較。
5: ifnull 41
這就是我在Java源代碼里書寫的IF分支。如果IF分支里檢測的outer引用為null,則直接返回了。體現(xiàn)在字節(jié)碼就是,如果ifnull為true,則跳轉(zhuǎn)到第41行字節(jié)碼,即直接返回。
如果ifnull不為true,則繼續(xù)執(zhí)行下去。又將outer引用加載到棧上。
從字節(jié)碼的分析可以觀察到一個(gè)有趣的現(xiàn)象,再次看看我們的IF語句。
Java編譯時(shí),編譯器實(shí)際將其轉(zhuǎn)換成了下面的寫法:
if (outer == null ) return; if( outer.nested == null ) return; if( outer.nested.inner == null) return; System.out.println(outer.nested.inner.foo);
這個(gè)事實(shí)可以通過下圖得到確認(rèn)。
javap生成的字節(jié)碼里的LineNumberTable也很有用。這張表里每行的line后面的數(shù)字代表Java源代碼的序號,line XX冒號后面的數(shù)字代表字節(jié)碼里每行指令的序號??纯聪聢D中Java源代碼和對應(yīng)的字節(jié)指令在LineNumberTable中的映射關(guān)系。
LineNumberTable維護(hù)了Java源代碼同字節(jié)指令的映射關(guān)系,確保了Java代碼調(diào)試的順利進(jìn)行。
要獲取更多Jerry的原創(chuàng)技術(shù)文章,請關(guān)注公眾號"汪子熙"或者掃描下面二維碼:
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/71747.html
摘要:我下圖代碼第五行和第九行分別定義了一個(gè)整型變量和一個(gè)整型常量程序員都知道兩者的區(qū)別。下面我們就用將文件反編譯出來然后深入研究里整型變量和整型常量的區(qū)別。 我下圖代碼第五行和第九行分別定義了一個(gè)整型變量和一個(gè)整型常量: static final int number1 = 512; static int number3 = 545; Java程序員都知道兩者的區(qū)別。 showImg(ht...
摘要:使用命令行將包含了這行代碼的類反編譯查看其字節(jié)碼我們看到字符串被編譯器加到了常量池里。代碼被翻譯成了下面兩句字節(jié)碼首先底層的原生方法被調(diào)用,生成的內(nèi)部存儲實(shí)現(xiàn)。做一個(gè)字符串拼接的操作。 我們看這樣一行簡單的字符串賦值操作的Java代碼。 String a = i042416; 使用命令行將包含了這行代碼的Java類反編譯查看其字節(jié)碼: javap -v constant.Constan...
摘要:但是有一個(gè)的指令,可以把字節(jié)碼翻譯成人類能看懂的東西。是文件分解器,可以反編譯即對編譯的文件進(jìn)行反編譯,也可以查看編譯器生成的字節(jié)碼?,F(xiàn)在有一個(gè)類,定義入下先用編譯成字節(jié)碼,再使用進(jìn)行反編譯。 概要 Java工程師面試官偏愛的問題之一,就是abc和 new String(abc)的區(qū)別是什么?回答的比較好的會帶出Java堆,棧,常量池,引用等概念。但今天不止如此,我們從指令的角度,去看...
摘要:反匯編器與反編譯器不同,反編譯器的目標(biāo)是高級語言而非匯編語言。反匯編器的反匯編輸出通常格式化為適合人類閱讀,而非用作匯編器的輸入源,因此它主要是一個(gè)逆向工程工具。本文章參考了通過命令分析匯編指令反匯編器 問題描述 寫這篇文章是為了記錄我這幾天遇到的一個(gè)疑惑,并且順藤摸瓜的學(xué)習(xí)一下javap命令。遇到的疑惑是這樣的:我在看使用枚舉類型實(shí)現(xiàn)單列模式的博客時(shí),發(fā)現(xiàn)一些博客中寫到的枚舉類型的反...
閱讀 2432·2023-04-26 00:46
閱讀 593·2023-04-25 21:36
閱讀 737·2021-11-24 10:19
閱讀 2282·2021-11-23 09:51
閱讀 1028·2021-10-21 09:39
閱讀 841·2021-09-22 10:02
閱讀 1677·2021-09-03 10:29
閱讀 2707·2019-08-30 15:53