深入java虛擬機(一)——java虛擬機底層結構詳解

来源:http://www.cnblogs.com/cao1234554321/archive/2016/01/12/5123572.html
-Advertisement-
Play Games

在以前的博客裡面,我們介紹了在java領域中大部分的知識點,從最基礎的java最基本語法到SSH框架。這裡面應該包含了在java領域裡面的大部分內容了吧。但是,那些知識點是讓我們從一個應用的層面上瞭解了java,java程式真正底層的運行機制和一些底層虛擬機的工作我們還不瞭解,雖然這些內容在我們真正...


在以前的博客裡面,我們介紹了在java領域中大部分的知識點,從最基礎的java最基本語法到SSH框架。這裡面應該包含了在java領域裡面的大部分內容了吧。但是,那些知識點是讓我們從一個應用的層面上瞭解了javajava程式真正底層的運行機制和一些底層虛擬機的工作我們還不瞭解,雖然這些內容在我們真正的開發中幾乎用不到這些底層的東西,但對於我們對java的理解會有比較大的幫助。尤其也對以後java開發中的性能優化有很大幫助,可以使我們減少一些沒必要的記憶體浪費等好處。所以,從今天開始,我將和大家一起來學習一下java虛擬機的內容。從底層開一下java的運行機制。

Java虛擬機

Java虛擬機(Java Virtual Machine) 簡稱JVM Java虛擬機是一個想象中的機器,在實際的電腦上通過軟體模擬來實現。Java虛擬機有自己想象中的硬體,如處理器、堆棧、寄存器等,還具有相應的指令系統。下麵我們就來看一下這幾部分比較重要的java虛擬機的結構

JVM寄存器

所有的CPU均包含用於保存系統狀態和處理器所需信息的寄存器組。如果虛擬機定義義較多的寄存器,便可以從中得到更多的信息而不必對棧或記憶體進行訪問,這有利於提高運行速度。然而,如果虛擬機中的寄存器比實際CPU的寄存器多,在實現虛擬機時就會占用處理器大量的時間來用常規存儲器模擬寄存器,這反而會降低虛擬機的效率。針對這種情況,JVM只設置了4個最為常用的寄存器。它們是:pc程式計數器,optop操作數棧頂指針 ,frame當前執行環境指針, vars指向當前執行環境中第一個局部變數的指針, 所有寄存器均為32位。pc用於記錄程式的執行。optop,framevars用於記錄指向Java棧區的指針。

JVM棧結構

作為基於棧結構的電腦,Java棧是JVM存儲信息的主要方法。當JVM得到一個java位元組碼應用程式後,便為該代碼中一個類的每一個方法創建一個棧框架,以保存該方法的狀態信息。每個棧框架包括以下三類信息:局部變數執行環境操作數棧 局部變數用於存儲一個類的方法中所用到的局部變數。vars寄存器指向該變數表中的第一個局部變數。執行環境用於保存解釋器對Java位元組碼進行解釋過程中所需的信息。它們是:上次調用的方法、局部變數指針和操作數棧的棧頂和棧底指針。執行環境是一個執行一個方法的控制中心。例如:如果解釋器要執行iadd(整數加法),首先要從frame寄存器中找到當前執行環境,而後便從執行環境中找到操作數棧,從棧頂彈出兩個整數進行加法運算,最後將結果壓入棧頂。  操作數棧用於存儲運算所需操作數及運算的結果。

JVM碎片回收堆

Java類的實例所需的存儲空間是在堆上分配的。解釋器具體承擔為類實例分配空間的工作。解釋器在為一個實例分配完存儲空間後,便開始記錄對該實例所占用的記憶體區域的使用。一旦對象使用完畢,便將其回收到堆中。在Java語言中,除了new語句外沒有其他方法為一對象申請和釋放記憶體。對記憶體進行釋放和回收的工作是由Java運行系統承擔的。這允許Java運行系統的設計者自己決定碎片回收的方法。在SUN公司開發的Java解釋器和Hot Java環境中,碎片回收用後臺線程的方式來執行。這不但為運行系統提供了良好的性能,而且使程式設計人員擺脫了自己控制記憶體使用的風險。

JVM存儲區

  JVM有兩類存儲區:常量緩衝池和方法區。常量緩衝池用於存儲類名稱、方法和欄位名稱以及串常量。方法區則用於存儲Java方法的位元組碼。對於這兩種存儲區域具體實現方式在JVM規格中沒有明確規定。這使得Java應用程式的存儲佈局必須在運行過程中確定,依賴於具體平臺的實現方式。JVM是為Java位元組碼定義的一種獨立於具體平臺的規格描述,是Java平臺獨立性的基礎。目前的JVM還存在一些限制和不足,有待於進一步的完善,但無論如何,JVM的思想是成功的。對比分析:如果把Java原程式想象成我們的C++原程式,Java原程式編譯後生成的位元組碼就相當於C++原程式編譯後的80x86的機器碼(二進位程式文件),JVM虛擬機相當於80x86電腦系統,Java解釋器相當於80x86CPU。在80x86CPU上運行的是機器碼,在Java解釋器上運行的是Java位元組碼。  Java解釋器相當於運行Java位元組碼的CPU,但該CPU不是通過硬體實現的,而是用軟體實現的。Java解釋器實際上就是特定的平臺下的一個應用程式。只要實現了特定平臺下的解釋器程式,Java位元組碼就能通過解釋器程式在該平臺下運行,這是Java跨平臺的根本。當前,並不是在所有的平臺下都有相應Java解釋器程式,這也是Java並不能在所有的平臺下都能運行的原因,它只能在已實現了Java解釋器程式的平臺下運行。

Java虛擬機的體繫結構圖

 

             

Java虛擬機從啟動到結束的生命周期,當java虛擬機啟動後,在如下幾種情況下,Java虛擬機將結束生命周期:

1.執行了System.exit()方法

2.程式正常執行結束

3.程式在執行過程中遇到了異常或錯誤而異常終止

4.由於操作系統出現錯誤而導致Java虛擬機進程終止

 

Java虛擬機的棧有三個區域:局部變數區、運行環境區、操作數區。

 

局部變數區

每個Java方法使用一個固定大小的局部變數集。它們按照與vars寄存器的字偏移量來定址。局部變數都是32位的。長整數和雙精度浮點數占據了兩個局部變數的空間,卻按照第一個局部變數的索引來定址。(例如,一個具有索引n的局部變數,如果是一個雙精度浮點數,那麼它實際占據了索引nn+1所代表的存儲空間)虛擬機規範並不要求在局部變數中的64位的值是64位對齊的。虛擬機提供了把局部變數中的值裝載到操作數棧的指令,也提供了把操作數棧中的值寫入局部變數的指令。

運行環境區

在運行環境中包含的信息用於動態鏈接,正常的方法返回以及異常捕捉。

操作數棧區

機器指令只從操作數棧中取操作數,對它們進行操作,並把結果返回到棧中。選擇棧結構的原因是:在只有少量寄存器或非通用寄存器的機器(Intel486),也能夠高效地模擬虛擬機的行為。操作數棧是32位的。它用於給方法傳遞參數,並從方法接收結果,也用於支持操作的參數,並保存操作的結果。例如,iadd指令將兩個整數相加。相加的兩個整數應該是操作數棧頂的兩個字。這兩個字是由先前的指令壓進堆棧的。這兩個整數將從堆棧彈出、相加,並把結果壓回到操作數棧中。

每個原始數據類型都有專門的指令對它們進行必須的操作。每個操作數在棧中需要一個存儲位置,除了longdouble,它們需要兩個位置。操作數只能被適用於其類型的操作符所操作。例如,壓入兩個int類型的數,如果把它們當作是一個long類型的數則是非法的。在Sun的虛擬機實現中,這個限制由位元組碼驗證器強制實行。但是,有少數操作(操作符dupeswap),用於對運行時數據區進行操作時是不考慮類型的。

本地方法棧,當一個線程調用本地方法時,它就不再受到虛擬機關於結構和安全限制方面的約束,它既可以訪問虛擬機的運行期數據區,也可以使用本地處理器以及任何類型的棧。例如,本地棧是一個C語言的棧,那麼當C程式調用C函數時,函數的參數以某種順序被壓入棧,結果則返回給調用函數。在實現Java虛擬機時,本地方法介面使用的是C語言的模型棧,那麼它的本地方法棧的調度與使用則完全與C語言的棧相同。

 

 

 

 

 

下圖可以表示出來java程式運行的一個全過程

  

 

3  Java虛擬機的運行過程

上面對虛擬機的各個部分進行了比較詳細的說明,下麵通過一個具體的例子來分析它的運行過程。

虛擬機通過調用某個指定類的方法main啟動,傳遞給main一個字元串數組參數,使指定的類被裝載,同時鏈接該類所使用的其它的類型,並且初始化它們。例如對於程式:

class HelloApp

{

    public static void main(String[] args)

    {

        System.out.println("Hello World!");

        for (int i = 0; i < args.length; i++ )

        {

            System.out.println(args[i]);

        }

    }

}

編譯後在命令行模式下鍵入: java HelloApp run virtual machine

將通過調用HelloApp的方法main來啟動java虛擬機,傳遞給main一個包含三個字元串"run""virtual""machine"的數組。現在我們略述虛擬機在執行HelloApp時可能採取的步驟。

開始試圖執行類HelloAppmain方法,發現該類並沒有被裝載,也就是說虛擬機當前不包含該類的二進位代表,於是虛擬機使用ClassLoader試圖尋找這樣的二進位代表。如果這個進程失敗,則拋出一個異常。類被裝載後同時在main方法被調用之前,必須對類HelloApp與其它類型進行鏈接然後初始化。鏈接包含三個階段:檢驗,準備和解析。檢驗檢查被裝載的主類的符號和語義,準備則創建類或介面的靜態域以及把這些域初始化為標準的預設值,解析負責檢查主類對其它類或介面的符號引用,在這一步它是可選的。類的初始化是對類中聲明的靜態初始化函數和靜態域的初始化構造方法的執行。一個類在初始化之前它的父類必須被初始化。整個過程如下:

     

 

 

參考資料:http://blog.csdn.net/csh624366188


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一:商城框架搭建示例圖二:文件存放目錄位置圖片 三:代碼部分index.php代碼1 host."";init.php代碼 1 autoExecute('user',array('username'=>'zs', 'email'=>'[email protected]', 'insert'));46 ...
  • 絕大多數的設計模式文章都介紹了command模式,但只涉及了文字解釋、結構圖和代碼實現。 只是在最終說了一下使用command模式能夠實現事務。 可是,你們到底是舉個例子啊。 可你們不舉。 你們不舉啊。 只能由我來舉個例子了。
  • 這段時間一直比較忙,一忙起來真感覺自己就只是一臺掙錢的機器了(說的好像能掙到多少錢似的,呵呵);這會兒難得有點兒空閑時間,想把前段時間開發微信公眾號支付遇到問題及解決方法跟大家分享下,這些“暗坑”能不掉就不掉吧,要不然關鍵時刻出問題,真是讓人急的焦頭爛額。 雙12客戶的商城活動正在蓄勢進行...
  • 我們有時候在工作流開發中可能會遇到這樣的需求,就是已經審批結束的流程,可能我們還是仍然需要修改業務表的結果,而且我們需要一個時間期限,比如:在5天內可以進行修改,這個時候我們就需要得到我們最後一步審批的時間,我們可以通過下麵這個sql查詢到該時間SELECT MAX(COMM.TIME_) FRO....
  • Python3中int、float、str、list、dict、tuple類的內建方法
  • 本小節我們來做一個好玩的事情,就是計數器,還記得在做LED自加實驗時我們就曾經提到過關於計數器的相關議題,那麼這節我們就來討論討論。 探討一下如下的問題:請用verilog記八個數的寫法,分析這個可以更好的理解觸發器的工作原理。1. reg [3:0]cnt; always@(pose...
  • Sub Macro2()'' Macro2 Macro'' Keyboard Shortcut: Ctrl+d' ActiveCell.Select ActiveSheet.Paste Selection.ShapeRange.ScaleHeight 0.6, msoFalse, msoScaleF...
  • 理解什麼是繼承首先我們知道,面對對象有三大特征:封裝:解決了數據的安全性問題繼承:解決了代碼的重用問題多態:解決了程式的擴展問題上一篇博客中,我們瞭解了一下封裝,現在我了再來看看什麼是繼承。在現實生活中,我們可以把封裝理解成兒子對父親財產的繼承。而在面向對象程式設計中的繼承,是一個對象從另一個對象獲...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...