一個簡單的例子教會您使用javap
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:\code\1.txt
檢視方法way1()對應的位元組碼:
下面這個wiki包含了java位元組碼裡每個指令的具體說明:
https://en.wikipedia.org/wiki/Java_bytecode_instruction_listings
下面對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的原創技術文章,請關注公眾號"汪子熙"或者掃描下面二維碼:
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/24475491/viewspace-2212656/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- 一個簡單例子教會你C++動態庫的用法單例C++
- 使用javap分析Java位元組碼的一個例子Java
- 一個簡單的「IOC」例子
- 擼一個簡單的MVVM例子MVVM
- 一個最簡單的 Github workflow 例子Github
- 一個簡單的netty通訊的例子Netty
- JUnit概述及一個簡單例子單例
- WebRTC:一個視訊聊天的簡單例子Web單例
- 一個簡單的生活例子,感受TRIZ的魅力!
- spring 簡單的使用 Hikari連線池 和 jdbc連線mysql 的一個簡單例子SpringJDBCMySql單例
- 一個簡單的例子瞭解async跟defer
- Unity如何連線伺服器: 一個簡單的例子Unity伺服器
- 簡單的理解Vuex,手把手教會Vue
- python+flask 編寫一個簡單的登入介面例子PythonFlask
- python+flask編寫一個簡單的登入介面例子PythonFlask
- SAP MM採購定價過程的一個簡單例子單例
- 一個簡單的例子理解Kubernetes的三種IP地址型別型別
- 通過一個簡單的例子,瞭解 Cypress 的執行原理
- SAP人工智慧服務Recast.AI的一個簡單例子人工智慧ASTAI單例
- Spring Boot 中使用Caffeine快取的簡單例子Spring Boot快取單例
- Rust 程式設計中使用 leveldb 的簡單例子Rust程式設計單例
- 簡單的整合 shiro + SpringMVC 例子SpringMVC
- 教小師妹學多執行緒,一個有深度的例子!執行緒
- Laravel 關聯查詢 ——一對一 簡單例子Laravel單例
- 怎樣錄製一個簡單的教學影片並配音講解
- Matplotlib1.簡單例子單例
- go語言如何入門?從一個簡單例子開始學起Go單例
- 使用 Angular Transfer State 的一個具體例子Angular
- RxJS CombineLatest operator 的一個具體使用例子JS
- Angular Reactive Form 的一個具體使用例子AngularReactORM
- 轉一篇OpenSSL的例子:簡單的TLS伺服器TLS伺服器
- 使用node啟動一個簡單的服務
- 使用nodeJS寫一個簡單的小爬蟲NodeJS爬蟲
- 使用 Python 構建一個簡單的 RESTful APIPythonRESTAPI
- 從一個例子中體會React的基本面React
- 教您使用IntelliJ IDEA鍵盤快捷鍵~IntelliJIdea
- 使用javap -v 命令遇到的問題Java
- Laravel 5 關聯查詢 ——一對 N 簡單例子Laravel單例