自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Java字節(jié)碼深入解析

開發(fā) 后端
本文將對Java字節(jié)碼進行解析,以及一些效率問題。

一:Java字節(jié)代碼的組織形式

類文件{

OxCAFEBABE,小版本號,大版本號,常量池大小,常量池數(shù)組,訪問控制標記,當前類信息,父類信息,實現(xiàn)的接口個數(shù),實現(xiàn)的接口信息數(shù)組,域個數(shù),域信息數(shù)組,方法個數(shù),方法信息數(shù)組,屬性個數(shù),屬性信息數(shù)組

}

二:查看方法 --- javap命令

例子:有一個Java類Demo.java

  1. public class Demo { 
  2.     private String str1; 
  3.     private String str2; 
  4.     private int num1; 
  5.     private int num2; 
  6.     public static final String STATIC_DATA = "hello world"
  7.      
  8.     private void sayHello1(){ 
  9.         System.out.println("this is method1..."); 
  10.     } 
  11.     private void sayHello2(){ 
  12.         System.out.println("this is method2..."); 
  13.     } 
  14.     public void sayHello3(){ 
  15.         System.out.println("this is method3..."); 
  16.     } 

通過jdk自帶的反編譯工具命令 javap 可以查看class文件的字節(jié)碼信息

D:\>javap -verbose Demo >> Demo.txt

Demo.txt:

  1. Compiled from "Demo.java" 
  2. public class Demo extends java.lang.Object 
  3.   SourceFile: "Demo.java" 
  4.   minor version: 0 
  5.   major version: 49   
  6.   
  7.   Constant pool: 
  8. const #1 = class      #2;   //  Demo 
  9. const #2 = Asciz     Demo; 
  10. const #3 = class      #4;   //  java/lang/Object 
  11. const #4 = Asciz     java/lang/Object; 
  12. const #5 = Asciz     str1; 
  13. const #6 = Asciz     Ljava/lang/String;; 
  14. const #7 = Asciz     str2; 
  15. const #8 = Asciz     num1; 
  16. const #9 = Asciz     I; 
  17. const #10 = Asciz   num2; 
  18. const #11 = Asciz   STATIC_DATA; 
  19. const #12 = Asciz   ConstantValue; 
  20. const #13 = String  #14//  hello world 
  21. const #14 = Asciz   hello world; 
  22. const #15 = Asciz   <init>; 
  23. const #16 = Asciz   ()V; 
  24. const #17 = Asciz   Code; 
  25. const #18 = Method       #3.#19;   //  java/lang/Object."<init>":()V 
  26. const #19 = NameAndType    #15:#16;//  "<init>":()V 
  27. const #20 = Asciz   LineNumberTable; 
  28. const #21 = Asciz   LocalVariableTable; 
  29. const #22 = Asciz   this
  30. const #23 = Asciz   LDemo;; 
  31. const #24 = Asciz   sayHello1; 
  32. const #25 = Field   #26.#28;  //  java/lang/System.out:Ljava/io/PrintStream; 
  33. const #26 = class    #27//  java/lang/System 
  34. const #27 = Asciz   java/lang/System; 
  35. const #28 = NameAndType    #29:#30;//  out:Ljava/io/PrintStream; 
  36. const #29 = Asciz   out; 
  37. const #30 = Asciz   Ljava/io/PrintStream;; 
  38. const #31 = String  #32//  this is method1... 
  39. const #32 = Asciz   this is method1...; 
  40. const #33 = Method       #34.#36;  //  java/io/PrintStream.println:(Ljava/lang/String;)V 
  41. const #34 = class    #35//  java/io/PrintStream 
  42. const #35 = Asciz   java/io/PrintStream; 
  43. const #36 = NameAndType    #37:#38;//  println:(Ljava/lang/String;)V 
  44. const #37 = Asciz   println; 
  45. const #38 = Asciz   (Ljava/lang/String;)V; 
  46. const #39 = Asciz   sayHello2; 
  47. const #40 = String  #41//  this is method2... 
  48. const #41 = Asciz   this is method2...; 
  49. const #42 = Asciz   sayHello3; 
  50. const #43 = String  #44//  this is method3... 
  51. const #44 = Asciz   this is method3...; 
  52. const #45 = Asciz   SourceFile; 
  53. const #46 = Asciz   Demo.java; 
  54.   
  55. public static final java.lang.String STATIC_DATA; 
  56.   Constant value: String hello world 
  57. public Demo(); 
  58.   Code: 
  59.    Stack=1, Locals=1, Args_size=1 
  60.    0:      aload_0 
  61.    1:      invokespecial  #18//Method java/lang/Object."<init>":()V 
  62.    4:      return 
  63.   LineNumberTable: 
  64.    line 20 
  65.   LocalVariableTable: 
  66.    Start  Length  Slot  Name   Signature 
  67.    0      5      0    this       LDemo; 
  68.   
  69. public void sayHello3(); 
  70.   Code: 
  71.    Stack=2, Locals=1, Args_size=1 
  72.    0:      getstatic   #25//Field java/lang/System.out:Ljava/io/PrintStream; 
  73.    3:      ldc   #43//String this is method3... 
  74.    5:      invokevirtual  #33//Method java/io/PrintStream.println:(Ljava/lang/String;)V 
  75.    8:      return 
  76.   LineNumberTable: 
  77.    line 170 
  78.    line 188 
  79.   LocalVariableTable: 
  80.    Start  Length  Slot  Name   Signature 
  81.    0      9      0    this       LDemo; 

解析:

1.版本號 major version: 49 //java版本 jdk1.6顯示的是50, jdk1.5顯示的是49,jdk1.4顯示的是58 , 高版本能執(zhí)行低版本的class文件

2.常量池Constant pool

Method:方法

Field:字段

String:字符串

Asciz:簽名如<init>由jvm調(diào)用,其他是不能夠去調(diào)用它的

NameAndType:變量名的類型

Class:類

通過字節(jié)碼,我們可以看到Demo類 繼承于java.lang.Object,如果類中沒有顯式聲明構(gòu)造函數(shù)的話,編譯器會插入一個缺省無參的構(gòu)造函數(shù)(構(gòu)造函數(shù)在JVM級別是顯示成<init>的普通函數(shù))。

三:檢測代碼的效率問題

學習Java的過程中,都會了解到字符串合并時要用到StringBuffer 來代替String,那下面就來通過Java字節(jié)碼來驗證兩種方式的效率性。

例子:一個Java類 TestString.java

  1. <strong>public class TestString { 
  2.     public String testString(String str1, String str2){ 
  3.        return str1 + str2; 
  4.     } 
  5.     public String testStringBuffer(StringBuffer sb, String str){ 
  6.        return sb.append(str).toString(); 
  7.     } 
  8.  </strong> 

javap –c TestString 后字節(jié)碼信息:

  1. Compiled from "TestString.java" 
  2. public class TestString extends java.lang.Object{ 
  3. public TestString(); 
  4.   Code: 
  5.    0:      aload_0 
  6.    1:      invokespecial  #8//Method java/lang/Object."<init>":()V 
  7.    4:      return 
  8.   
  9. public java.lang.String testString(java.lang.String, java.lang.String); 
  10.   Code: 
  11.    0:      new #16//class java/lang/StringBuilder 
  12.    3:      dup 
  13.    4:      aload_1 
  14.    5:      invokestatic    #18//Method java/lang/String.valueOf:(Ljava/lang/Object;)Ljava/lang/String; 
  15.    8:      invokespecial  #24//Method java/lang/StringBuilder."<init>":(Ljava/lang/String;)V 
  16.    11:     aload_2 
  17.    12:    invokevirtual  #27//Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 
  18.    15:    invokevirtual  #31//Method java/lang/StringBuilder.toString:()Ljava/lang/String; 
  19.    18:    areturn 
  20.   
  21. public java.lang.String testStringBuffer(java.lang.StringBuffer, java.lang.String); 
  22.   Code: 
  23.    0:      aload_1 
  24.    1:      aload_2 
  25.    2:      invokevirtual  #40//Method java/lang/StringBuffer.append:(Ljava/lang/String;)Ljava/lang/StringBuffer; 
  26.    5:      invokevirtual  #45//Method java/lang/StringBuffer.toString:()Ljava/lang/String; 
  27.    8:      areturn 

從上面編譯后的字節(jié)碼信息可以看出來,方法testString 調(diào)用了五個方法:new 、invokestatic 、invokespecial 和兩個invokevirtual ; 而testStringBuffer 方法只調(diào)用了兩個invokevirtual 方法。***個方法比第二個方法多做了好多工作,其效率當然是要低的。而且我們從java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;

可以看出來其實對于String字符串合并,內(nèi)部還是轉(zhuǎn)化為StringBuilder的方法調(diào)用,這是因為String是長度不可變的,所以不如直接采用StringBuilder(與StringBuffer 長度都是可變的,只不過前者是非線程安全,后者是線程安全)進行字符串合并。

原文鏈接:http://15838341661-139-com.iteye.com/blog/1287866

【編輯推薦】

  1. 解析Java語言11個主要特性
  2. Java中Date各種相關(guān)用法
  3. 如何實現(xiàn)高效的 JavaScript 單元測試
  4. Java虛擬機及JVM體系結(jié)構(gòu)
  5. Java語言的XPath API
責任編輯:林師授 來源: 周凡楊的博客
相關(guān)推薦

2021-12-09 22:36:30

Java 字節(jié)碼頁緩存

2010-09-25 10:20:05

JAVA字節(jié)碼

2016-10-31 19:41:29

Java垃圾回收

2022-01-17 11:28:55

JVM 虛擬機Java

2009-03-16 15:47:16

Java線程多線程

2024-10-21 10:45:52

2023-07-03 08:11:48

java字節(jié)碼字段

2019-10-30 08:45:21

JS代碼NodeJS

2022-03-30 10:10:17

字節(jié)碼??臻g

2019-12-20 12:38:28

Java技術(shù)工具

2016-05-18 17:15:17

互動出版網(wǎng)

2020-10-20 10:17:20

Java泛型Type

2017-02-07 09:54:43

JVMJavaClass

2023-08-30 11:03:47

Java工具

2012-01-12 09:20:49

Java

2013-11-26 16:32:47

Android關(guān)機移動編程

2010-09-17 15:44:21

網(wǎng)絡協(xié)議

2010-10-09 11:20:13

2018-04-04 15:05:17

虛擬機字節(jié)碼引擎

2012-03-28 10:30:33

ScalaJava
點贊
收藏

51CTO技術(shù)棧公眾號