天天看點

面試官:什麼是位元組碼?它最大的優勢是什麼?

關注“Java後端技術全棧”

回複“面試”擷取全套面試資料

什麼是位元組碼?
這個問題,面試官可以衍生提問,Java 是編譯執行的語言,還是解釋執行的語言。

Java 中引入了虛拟機的概念,即在機器和編譯程式之間加入了一層抽象的虛拟的機器。這台虛拟的機器在任何平台上都提供給編譯程式一個的共同的接口。

編譯程式隻需要面向虛拟機,生成虛拟機能夠了解的代碼,然後由解釋器來将虛拟機代碼轉換為特定系統的機器碼執行。在 Java 中,這種供虛拟機了解的代碼叫做位元組碼(即擴充名為 

.class

 的檔案),它不面向任何特定的處理器,隻面向虛拟機。

随便找一個項目彙總的.class檔案,然後使用16進制的方法檢視:

0000000: cafe babe 0000 0034 0061 0a00 1600 4709  .......4.a....G.
0000010: 0005 0048 0900 0500 4909 0005 004a 0700  ...H....I....J..
0000020: 4b0a 0005 004c 0a00 0500 4d0a 0016 004e  K....L....M....N
0000030: 0a00 0500 4f0a 0005 0050 0a00 1600 5107  ....O....P....Q.
0000040: 0052 0a00 0c00 4708 0053 0a00 0c00 540a  .R....G..S....T.
0000050: 000c 0055 0800 5608 0057 0a00 0c00 5808  ...U..V..W....X.
0000060: 0059 0a00 0c00 5a07 005b 0700 5c01 0005  .Y....Z..[..\...
0000070: 7461 6749 6401 0013 4c6a 6176 612f 6c61  tagId...Ljava/la
0000080: 6e67 2f49 6e74 6567 6572 3b01 0007 7461  ng/Integer;...ta
0000090: 674e 616d 6501 0012 4c6a 6176 612f 6c61  gName...Ljava/la
00000a0: 6e67 2f53 7472 696e 673b 0100 0574 6f74  ng/String;...tot
00000b0: 616c 0100 0149 0100 063c 696e 6974 3e01  al...I...<init>.
00000c0: 0003 2829 5601 0004 436f 6465 0100 0f4c  ..()V...Code...L00000d0: 696e 654e 756d 6265 7254 6162 6c65 0100  ineNumberTable..
00000e0: 124c 6f63 616c 5661 7269 6162 6c65 5461  .LocalVariableTa
00000f0: 626c 6501 0004 7468 6973 0100 1f4c 636f  ble...this...Lco
0000100: 6d2f 6a61 7661 2f74 6961 6e2f 626c 6f67  m/java/tian/blog
0000110: 2f65 6e74 6974 792f 5461 673b 0100 0867  /entity/Tag;...g
0000120: 6574 5461 6749 6401 0015 2829 4c6a 6176  etTagId...()Ljav
0000130: 612f 6c61 6e67 2f49 6e74 6567 6572 3b01  a/lang/Integer;.0000140: 000a 6765 7454 6167 4e61 6d65 0100 1428  ..getTagName...(0000150: 294c 6a61 7661 2f6c 616e 672f 5374 7269  )Ljava/lang/Stri0000160: 6e67 3b01 0008 6765 7454 6f74 616c 0100  ng;...getTotal..0000170: 0328 2949 0100 0873 6574 5461 6749 6401  .()I...setTagId.0000180: 0016 284c 6a61 7661 2f6c 616e 672f 496e  ..(Ljava/lang/In
0000190: 7465 6765 723b 2956 0100 104d 6574 686f  teger;)V...Metho
00001a0: 6450 6172 616d 6574 6572 7301 000a 7365  dParameters...se
00001b0: 7454 6167 4e61 6d65 0100 1528 4c6a 6176  tTagName...(Ljav00001c0: 612f 6c61 6e67 2f53 7472 696e 673b 2956  a/lang/String;)V
00001d0: 0100 0873 6574 546f 7461 6c01 0004 2849  ...setTotal...(I
00001e0: 2956 0100 0665 7175 616c 7301 0015 284c  )V...equals...(L00001f0: 6a61 7661 2f6c 616e 672f 4f62 6a65 6374  java/lang/Object
0000200: 3b29 5a01 0001 6f01 0012 4c6a 6176 612f  ;)Z...o...Ljava/
0000210: 6c61 6e67 2f4f 626a 6563 743b 0100 056f  lang/Object;...o
0000220: 7468 6572 0100 0a74 6869 7324 7461 6749  ther...this$tagI
0000230: 6401 000b 6f74 6865 7224 7461 6749 6401  d...other$tagId.
0000240: 000c 7468 6973 2474 6167 4e61 6d65 0100  ..this$tagName..
0000250: 0d6f 7468 6572 2474 6167 4e61 6d65 0100  .other$tagName..
0000260: 0d53 7461 636b 4d61 7054 6162 6c65 0700  .StackMapTable..
0000270: 4b07 005b 0100 0863 616e 4571 7561 6c01  K..[...canEqual.
0000280: 0008 6861 7368 436f 6465 0100 0550 5249  ..hashCode...PRI
0000290: 4d45 0100 0672 6573 756c 7401 0006 2474  ME...result...$t
00002a0: 6167 4964 0100 0824 7461 674e 616d 6501  agId...$tagName.
00002b0: 0008 746f 5374 7269 6e67 0100 0a53 6f75  ..toString...Sou
00002c0: 7263 6546 696c 6501 0008 5461 672e 6a61  rceFile...Tag.ja
00002d0: 7661 0c00 1e00 1f0c 0018 0019 0c00 1a00  va..............
00002e0: 1b0c 001c 001d 0100 1d63 6f6d 2f6a 6176  .........com/jav
00002f0: 612f 7469 616e 2f62 6c6f 672f 656e 7469  a/tian/blog/enti
0000300: 7479 2f54 6167 0c00 3e00 330c 0025 0026  ty/Tag..>.3..%.&
0000310: 0c00 3200 330c 0027 0028 0c00 2900 2a0c  ..2.3..'.(..).*.
0000320: 003f 002a 0100 176a 6176 612f 6c61 6e67  .?.*...java/lang
0000330: 2f53 7472 696e 6742 7569 6c64 6572 0100  /StringBuilder..
0000340: 0a54 6167 2874 6167 4964 3d0c 005d 005e  .Tag(tagId=..].^
0000350: 0c00 5d00 5f01 000a 2c20 7461 674e 616d  ..]._..., tagNam
0000360: 653d 0100 082c 2074 6f74 616c 3d0c 005d  e=..., total=..]
25 lines filtered                                                
           

檢視方式是先使用vim将.class檔案打開,然後輸入

:%!xxd

然後就可以看到

cafe babe

開頭的位元組碼了。

另外一種方式檢視位元組碼的方式:

xxd Tag.class Tag.txt

和上面一樣。

二進制與16進制轉換還有其他一些方式,如下:

以十六進制格式輸出:

od [選項] 檔案

od -d 檔案  十進制輸出

-o 檔案  八進制輸出

-x 檔案  十六進制輸出

xxd 檔案  輸出十六進制

在vi指令狀态下:

:%!xxd   :%!od    将目前文本轉化為16進制格式

:%!xxd -c 12 每行顯示12個位元組

:%!xxd -r    将目前文本轉化回文本格式

上面的位元組碼看起來是不是很無語,很多人是對其很厭煩。其實也沒那麼難的。

每一種平台的解釋器是不同的,但是實作的虛拟機是相同的。Java 源程式經過編譯器編譯後變成位元組碼,位元組碼由虛拟機解釋執行,虛拟機将每一條要執行的位元組碼送給解釋器,解釋器将其翻譯成特定機器上的機器碼,然後在特定的機器上運作。這也就是解釋了 Java 的編譯與解釋并存的特點。

Java 源代碼
=> 編譯器 => JVM 可執行的 Java 位元組碼(即虛拟指令)
=> JVM => JVM 中解釋器 => 機器可執行的二進制機器碼 => 程式運作
           
采用位元組碼的好處?

Java 語言通過位元組碼的方式,在一定程度上解決了傳統解釋型語言執行效率低的問題,同時又保留了解釋型語言可移植的特點。是以 Java 程式運作時比較高效,而且,由于位元組碼并不專對一種特定的機器,是以,Java程式無須重新編譯便可在多種不同的計算機上運作。

解釋型語言:解釋型語言,是在運作的時候将程式翻譯成機器語言。解釋型語言的程式不需要在運作前編譯,在運作程式的時候才翻譯,專門的解釋器負責在每個語句執行的時候解釋程式代碼。這樣解釋型語言每執行一次就要翻譯一次,效率比較低。——百度百科

例如:Python、PHP 。

推薦閱讀

【原創】Spring Boot終極篇《上》

【原創】Spring Boot終極篇《下》

繼續閱讀