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