(一)、java的優點之一是與平臺的無關性,那它是怎么做到的呢?
Java語言與平臺的無關性是使用Java虛擬機(JVM)是實現這一特點主要原因所在。
一般的語言如果要在不同的平臺上運行,至少需要編譯成不同的目標代碼。而引入Java語言虛擬機后,Java語言在不同平臺上運行時不需要重新編譯。Java語言使用模式Java虛擬機屏蔽了與具體平臺相關的信息,使得Java語言編譯程序只需生成在Java虛擬機上運行的目標代碼(字節碼),就可以在多種平臺上不加修改地運行。Java虛擬機在執行字節碼時,把字節碼解釋成具體平臺上的機器指令執行。
所以JVM的重要性顯而易見。
(二)、JVM的整體架構是什么樣子呢?
主要包括兩個子系統和兩個組件:
(1)、 Class loader(類裝載器) 子系統;
(2)、Execution engine(執行引擎) 子系統;
(3)、Native interface(本地接口)組件;
(4)、Runtime data area (運行時數據區域)組件
結構圖如下:
<!–[endif]–>
(1)、 Class loader(類裝載器) 子系統;
根據給定的全限定名類名(如 java.lang.Object)來裝載class文件的內容到 Runtime data area中的method area(方法區域)。Javsa程序員可以extends java.lang.ClassLoader類來寫自己的Class loader。
(2)、Execution engine(執行引擎) 子系統;
執行classes中的指令。任何JVM specification實現(JDK)的核心是Execution engine, 換句話說:Sun的JDK 和IBM的JDK好壞主要取決于他們各自實現的Execution engine的好壞。每個運行中的線程都有一個Execution engine的實例。
(3)、Native interface(本地接口)組件;
Native interface與native libraries交互,是其它編程語言交互的接口。
(4)、Runtime data area (運行時數據區域)組件
運行時數據區域組件包含:Heap (堆)、 Method Area(方法區域)、Java Stack(java的棧)、PRogram Counter(程序計數器)、Native method stack(本地方法棧);
Heap 和Method Area是被所有線程的共享使用的;而Java stack, Program counter 和Native method stack是以線程為粒度的,每個線程獨自擁有。
Heap Java程序在運行時創建的所有類實或數組都放在同一個堆中。而一個Java虛擬實例中只存在一個堆空間,因此所有線程都將共享這個堆。每一個java程序獨占一個JVM實例,因而每個java程序都有它自己的堆空間,它們不會彼此干擾。但是同一java程序的多個線程都共享著同一個堆空間,就得考慮多線程訪問對象(堆數據)的同步問題。 (這里可能出現的異常java.lang.OutOfMemoryError: Java heap space)
Method Area 在Java虛擬機中,被裝載的class的信息存儲在Method area的內存中。當虛擬機裝載某個類型時,它使用類裝載器定位相應的class文件,然后讀入這個class文件內容并把它傳輸到虛擬機中。緊接著虛擬機提取其中的類型信息,并將這些信息存儲到方法區。該類型中的類(靜態)變量同樣也存儲在方法區中。與Heap一樣,method area是多線程共享的,因此要考慮多線程訪問的同步問題。比如,假設同時兩個線程都企圖訪問一個名為Lava的類,而這個類還沒有內裝載入虛擬機,那么,這時應該只有一個線程去裝載它,而另一個線程則只能等待。 (這里可能出現的異常java.lang.OutOfMemoryError: PermGen full)
Java棧(Java Stack)是線程私有的,它的生命周期與線程相同。Java stack以幀為單位保存線程的運行狀態。虛擬機只會直接對Java stack執行兩種操作:以幀為單位的壓棧或出棧。每當線程調用一個方法的時候,就對當前狀態作為一個幀保存到java stack中(壓棧);當一個方法調用返回時,從java stack彈出一個幀(出棧)。棧的大小是有一定的限制,這個可能出現StackOverFlow問題。
程序計數器(Program counter)是一塊較小的內存空間,它的作用可以看做是當前線程所在會想的字節碼的行號指示器。字節碼解釋器工作時就是通過改變這個計數器的值來選取下一條需要執行的字節碼指令、分支、循環、跳轉、異常處理線程恢復等基礎功能都需要依賴這個計數器來完成。 每個運行中的Java程序,每一個線程都有它自己的PC寄存器,也是該線程啟動時創建的。PC寄存器的內容總是指向下一條將被執行指令的地址,這里的地址可以是一個本地指針,也可以是在方法區中相對應于該方法起始指令的偏移量。
Native method stack 對于一個運行中的Java程序而言,它還能會用到一些跟本地方法相關的數據區。當某個線程調用一個本地方法時,它就進入了一個全新的并且不再受虛擬機限制的世界。本地方法可以通過本地方法接口來訪問虛擬機的運行時數據區,不止與此,它還可以做任何它想做的事情。比如,可以調用寄存器,或在操作系統中分配內存等。總之,本地方法具有和JVM相同的能力和權限。 (這里出現JVM無法控制的內存溢出問題native heap OutOfMemory );
新聞熱點
疑難解答