細說虛擬機棧

文章出自 

http://www.mynawang.com/

 

引文

    對JVM不瞭解的請先訪問文章《Java虛擬機運行時數據區》,相信會讓你大體瞭解各個內存區域的對應功能。接着,咱們開始細說虛擬機棧     前文中說到:「虛擬機棧是線程私有的,每建立一個線程,虛擬機就會爲這個線程建立一個虛擬機棧,虛擬機棧表示Java方法執行的內存模型,每調用一個方法就會爲每一個方法生成一個棧幀(Stack Frame),用來存儲局部變量表、操做數棧、動態連接、方法出口等信息。每一個方法被調用和完成的過程,都對應一個棧幀從虛擬機棧上入棧和出棧的過程。虛擬機棧的生命週期和線程是相同的」。數據結構

虛擬機棧

    其中,虛擬機棧是一個後入先出的棧。棧幀是保存在虛擬機棧中的,棧幀是用來存儲數據和存儲部分過程結果的數據結構,同時也被用來處理動態連接(Dynamic Linking)、方法返回值和異常分派(Dispatch Exception)。線程運行過程當中,只有一個棧幀是處於活躍狀態,稱爲「當前活躍棧幀」,當前活動棧幀始終是虛擬機棧的棧頂元素。以下圖所示: 棧流程優化

棧幀

    上述內容已對棧幀作了大體介紹,接下去仔細描述棧幀中的操做數棧,動態鏈接,方法返回地址和一些額外的附加信息。 以下圖所示:spa

棧幀詳情

1.局部變量表線程

    局部變量表是一組局部變量值存儲空間,用於存放方法參數和方法內部定義的局部變量。在Java文件編譯爲Class文件時,就在方法表的Code屬性的max_locals數據項中肯定了該方法須要分配的最大局部變量表的容量。code

2.操做數棧orm

    操做數棧也常被稱爲操做棧,它是一個後入先出棧。JVM底層字節碼指令集是基於棧類型的,全部的操做碼都是對操做數棧上的數據進行操做,對於每個方法的調用,JVM會創建一個操做數棧,以供計算使用。和局部變量同樣。操做數棧的最大深度也是編譯的時候寫入到方法表的code屬性的max_stacks數據項中。操做數棧的每個元素能夠是任意的Java數據類型,包括long、double。32位數據類型所佔的棧容量爲1,64位數據類型所佔的棧容量爲2。棧容量的單位爲「字寬」,對於32位虛擬機來講,一個「字寬」佔4個字節,64位虛擬機來講,一個「字寬」佔8個字節。當一個方法剛剛執行的時候,這個方法的操做數棧是空的,在方法執行的過程當中,會有各類字節碼指向操做數棧中寫入和提取值,也就是入棧與出棧操做。例如,在作算術運算的時候就是經過操做數棧來進行的,又或者調用其它方法的時候是經過操做數棧來行參數傳遞的。 另外,在概念模型中,兩個棧幀做爲虛擬機棧的元素,相互之間是徹底獨立的,可是大多數虛擬機的實現裏都會做一些優化處理,令兩個棧幀出現一部分重疊。讓下棧幀的部分操做數棧與上面棧幀的部分局部變量表重疊在一塊兒,這樣在進行方法調用返回時就能夠共用一部分數據,而無須進行額外的參數複製傳遞了。生命週期

3.動態鏈接內存

    每一個棧幀都包含一個指向運行時常量池中該棧幀所屬性方法的引用,持有這個引用是爲了支持方法調用過程當中的動態鏈接。在Class文件的常量池中存有大量的符號引用,字節碼中的方法調用指令就以常量池中指向方法的符號引用爲參數。這些符號引用一部分會在類加載階段或第一次使用的時候轉化爲直接引用,這種轉化稱爲靜態解析。另一部分將在每一次的運行期期間轉化爲直接引用,這部分稱爲動態鏈接開發

4.方法返回地址get

    當一個方法被執行後,有兩種方式退出這個方法。第一種方式是執行引擎遇到任意一個方法返回的字節碼指令,這時候可能會有返回值傳遞給上層的方法調用者(調用當前方法的的方法稱爲調用者),是否有返回值和返回值的類型將根據遇到何種方法返回指令來決定,這種退出方法方式稱爲正常完成出口(Normal Method Invocation Completion)。另一種退出方式是,在方法執行過程當中遇到了異常,而且這個異常沒有在方法體內獲得處理,不管是Java虛擬機內部產生的異常,仍是代碼中使用athrow字節碼指令產生的異常,只要在本方法的異常表中沒有搜索到匹配的異常處理器,就會致使方法退出,這種退出方式稱爲異常完成出口(Abrupt Method Invocation Completion)。一個方法使用異常完成出口的方式退出,是不會給它的調用都產生任何返回值的。     不管採用何種方式退出,在方法退出以前,都須要返回到方法被調用的位置,程序才能繼續執行,方法返回時可能須要在棧幀中保存一些信息,用來幫助恢復它的上層方法的執行狀態。通常來講,方法正常退出時,調用者PC計數器的值就能夠做爲返回地址,棧幀中極可能會保存這個計數器值。而方法異常退出時,返回地址是要經過異常處理器來肯定的,棧幀中通常不會保存這部分信息。 方法退出的過程實際上等同於把當前棧幀出棧,所以退出時可能執行的操做有:恢復上層方法的局部變量表和操做數棧,把返回值(若是有的話)壓入調用都棧幀的操做數棧中,調用PC計數器的值以指向方法調用指令後面的一條指令等。

5.附加信息

    虛擬機規範容許具體的虛擬機實現增長一些規範裏沒有描述的信息到棧幀中,例如與高度相關的信息,這部分信息徹底取決於具體的虛擬機實現。在實際開發中,通常會把動態鏈接,方法返回地址與其它附加信息所有歸爲一類,稱爲棧幀信息。

相關文章
相關標籤/搜索