成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

《深入理解Java虛擬機》筆記

開發 后端 開發工具
“編好的機器指令”當然指的是能在CPU上運行的,如果這里我還實現了一個翻譯機器:從自己定義的格式指令翻譯到CPU指令,那么就可以執行根據自定義格式的代碼了……

在C里面我們想執行一段自己編寫的機器指令的方法大概如下:

  1. typedef void(*FUNC)(int);  
  2. char* str = "your code";  
  3. FUNC f = (FUNC)str;  
  4. (*f)(0); 

也就是說,我們完全可以做一個工具,從一個文件中讀入指令,然后將這些指令運行起來。上面代碼中“編好的機器指令”當然指的是能在CPU上運行的,如果這里我還實現了一個翻譯機器:從自己定義的格式指令翻譯到CPU指令,那么就可以執行根據自定義格式的代碼了。那么上面這段代碼是不是相當于最簡單的一個虛擬機了?下面來看JVM的總體結構:

 

ClassLoader的作用是裝載能被JVM識別的指令(當然不只是從磁盤文件或內存去裝載),那么我們先了解一下該格式:

魔數以及版本就不說了(滿大街的文件格式都是這個東西),接著的便是常量池,其中無非是兩種東西:

  1. 字面常量(比如Integer、Long、String等);
  2. 符號引用(方法是哪里的?什么樣的?);

而我們知道,在JVM里面Class都是根據全限定名去找的,那么方法的描述當然也應該如此,那么就得到這些常量之間的關系如下:

  

在接下來的“訪問權限”中表明了該Class 是public還是private等,而this&super&interface則表面了“本類”、“繼承自哪個類”、“實現了哪些接口”,實際上這里只是保存了代表這些信息的CONSTANT_Class_info的下標(u2)。

  

感覺這里的NameIndex和 DescriptorIndex加起來和NameAndType有點像,那么為什么不直接用一個NameAndType的索引值表示?MethodInfo和FieldInfo之間最大的不同點就是Attributes。比如FieldInfo的屬性表中存放的是變量的初始值,而 MethodInfo的屬性表中存放的則是字節碼。那么我們來依次看這些Attributes,首先是Code:

  

有幾個有意思的地方:

  1. 從Class文件中可以知道在執行的過程中棧的深度;
  2. 對于非靜態方法,編譯器會將this通過參數傳遞給方法;
  3. 異常表中記錄的范圍是指令的行數(而不是源代碼的);
  4. 這里的異常是指try-catch中的,而與Code同級的異常表中的則是指throws出去的;

Exceptions則非常簡單:

LineNumberTable保存了字節碼和源碼之間的關系,結構如下:

  

LocalVariableTable描述了棧幀中局部變量表的變量和源代碼中定義的變量之間的關系,結構如下:

  

SourceFile指明了生成該Class文件的Java源碼文件名(比如在一個Java文件中申明了很多類的時候會生成很多Class文件),結構如下:

Deprecated和Synthetic屬性只存在“有”和“沒有”的區別:

  1. Deprecated:被程序作者定為不再推薦使用,通過@deprecated注釋說明;
  2. Synthetic:表示字段或方法是由編譯器自動生成的,比如<init>; 

這也就是為什么Code屬性后面會有Attribute的原因?

類加載的時機就很簡單了:在用到的時候就加載(廢話!)。下來看一下類加載的過程:

  

執行上面這段過程的是:ClassLoader,這個東西還是非常重要的,在JVM中是通過ClassLoader和類本身共同去判斷兩個Class是否相同。換句話說就是:不同的ClassLoader加載同一個Class文件,那么JVM認為他們生成的類是不同的。有些時候不會從Class文件中加載流(比如Java Applet是從網絡中加載),那么這個ClassLoader和普通的實現邏輯當然是不一樣的,通過不同的ClassLoader就可以解決這個問題。

但是允許使用不同的ClassLoader又引發了新的問題:如果我也聲明了一個java.lang.Integer,但是里面的代碼非常危險,怎么辦?這里就引出了雙親委派模式:

除了頂層的啟動類加載器外,其余的類加載器都應該有父類加載器(通過組合實現),它在接到加載類的請求時優先委派給父類加載器去完成。

這樣的話,在加載java.lang.Integer的時候會優先使用系統的類加載器,這樣就不會加載用戶自己寫的。在Java程序員看到有3種系統提供的類加載器:

  1. Bootstrap ClassLoader:負責加載<JAVA_HOME>\lib目錄中的類庫,無法被Java程序直接引用;
  2. Extension ClassLoader:負責加載<JAVA_HOME>\lib\ext,開發者可以直接使用;
  3. Application ClassLoader:加載ClassPath上所指定的類庫,如果沒有自己定義過自己的類加載器則會使用它;

這樣默認的類會是有Application ClassLoader去加載類,然后如果發現要使用新的類型的時候則會遞歸地使用Application ClassLoader去加載(在前面的加載過程中提到)。這樣,只有在自己的程序中能使用自己編寫的ClassLoader去加載類,并且這個被加載的類是不能被別人使用的。

雙親委派模式不是一個強制性的約束,而是Java設計者推薦給開發者的類加載實現方式。雙親委派模式出現過的3次“破壞”:

  1. 為了兼容JDK 1.0,建議使用者去覆蓋findClass方法;
  2. 在基礎類要訪問用戶類的代碼會出現問題(比如JNDI):線程山下文類加載器;
  3. 用戶的一些需求,比如HotSwap、OSGI等; 

加載完完成后,接下來就要看程序是怎么運行的。棧幀是用于支持虛擬機進行方法調用和執行,幀的意思就是一個單位,在調用其他方法的時候會向棧中壓入棧幀,結構如下:

  

在Class文件編譯完成之后,在運行的時候需要多少個局部變量就已經確定(在前面Class文件中也已經看到過了),那么這里需要注意這個特性可能會引發GC(具體如何引發就不在這里細說了)。在棧中,總是底層的棧去調用高層的棧(并且一定的相鄰的),那么他們在參數傳遞(返回結果)的時往往是通過將其壓入操作數棧,有些虛擬機為了提高這部分的效率使得相鄰棧幀“糾纏”在一起:

  

那么我們接下來要去看是方法是如何執行的,第一個問題就是執行哪個方法?在“面向過程”的編程中似乎不存在在個問題,但是在Java OR C++中這都是比較蛋疼的一個問題。原因就是平時不會這么用,但是你必須去搞明白= =。JVM確定目標方法的時候有兩種方法:

  1. 靜態分派:根據參數類型和方法名稱來決定調用哪個方法。但是,并不是說沒有發現匹配的類型就報錯,比如有:func(int a),而在調用func('a')的時候也會調用該方法(當然是在沒有func(char a)的前提下),這樣給人的關鍵就有點像一個處理的鏈條。不管多么復雜,這些都是在編譯期間確定的,因為這里是向上找的。
  2. 動態分派:最普遍的就是Interface a = new Implements(),a調用方法到底應該是哪個類的在編譯期間是無法確定的。其實動態分派實現起來也很簡單:在調用方法的時候先拿到對象的實際類型。

其實“靜態”和“動態”給人的感覺還是比較模糊的,“靜態分派”給人的感覺是根據參數的類型向上查找方法,“動態分派”給人的感覺則是根據實例的真實類型向上查找。虛擬機優化動態分派的效率一般是為類在方法區中建立一個虛方法表:

虛方法表中存放各個方法實際入口地址,如果某個方法在子類中沒有被重寫,那么子類的虛方法表里面的地址入口和父類相同方法的地址入口是一致的,都指向父類的實現入口。如果子類重寫了這個方法,子類方法表中的地址將會被替換為指向子類實現版本的入口地址。其實往簡單里說,就是一個預處理。

原文鏈接:http://www.cnblogs.com/tianchi/archive/2012/11/11/2761631.html

責任編輯:林師授 來源: 博客園
相關推薦

2024-03-29 11:42:21

Java虛擬機

2019-07-24 16:04:47

Java虛擬機并發

2024-04-03 13:49:00

Java虛擬機方法區

2016-09-01 12:37:13

OpenStack虛擬機Metadata

2024-03-26 07:30:07

Java虛擬機源文件

2023-09-22 23:00:11

Java虛擬機

2024-04-10 07:40:45

Java虛擬機內存

2021-05-13 21:27:24

ThreadLocal多線程多線程并發安全

2019-12-31 10:45:30

JavaVisualVM高并發

2011-12-28 13:24:47

JavaJVM

2011-12-28 13:38:00

JavaJVM

2010-06-01 15:25:27

JavaCLASSPATH

2016-12-08 15:36:59

HashMap數據結構hash函數

2020-07-21 08:26:08

SpringSecurity過濾器

2022-08-21 16:52:27

Linux虛擬內存

2011-03-17 09:58:43

Java虛擬機JVM

2012-03-05 11:09:01

JavaClass

2023-10-19 11:12:15

Netty代碼

2021-02-17 11:25:33

前端JavaScriptthis

2009-09-25 09:14:35

Hibernate日志
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 99久久婷婷国产综合精品电影 | 99久久婷婷国产综合精品首页 | 久久国产一区二区 | 蜜臀久久99精品久久久久野外 | 欧美一区二区三区在线观看 | 欧美精品1区2区 | 高清一区二区视频 | 国内自拍偷拍一区 | 不用播放器看的av | 国产盗摄视频 | 91精品国产综合久久婷婷香蕉 | 免费电影av| 欧美成人精品在线观看 | 免费看一区二区三区 | 久久国产精品一区二区三区 | 黄片毛片在线观看 | 91精品国产综合久久精品 | 日韩免费在线观看视频 | 亚洲综合一区二区三区 | 在线观看黄色电影 | 国产精品日韩一区二区 | 亚洲91av| 先锋av资源网 | 秋霞a级毛片在线看 | 午夜精品一区二区三区在线视频 | 91成人免费看片 | 国产在线一区二区三区 | 成人免费毛片在线观看 | 天天操天天玩 | av网站免费观看 | 国产日韩欧美精品一区二区三区 | 国产日韩欧美91 | 精品国产一区二区三区久久狼黑人 | 九九久久国产 | 欧美性一区二区三区 | 亚洲欧洲在线观看视频 | 91久久北条麻妃一区二区三区 | 天天操天天干天天曰 | 欧美精品片 | 超碰在线播 | 欧美精品一区二区在线观看 |