初步瞭解JVM第一篇

来源:https://www.cnblogs.com/linzepeng/archive/2019/12/15/12045148.html
-Advertisement-
Play Games

大家都知道,Java中JVM的重要性,學習了JVM你對Java的運行機制、編譯過程和如何對Java程式進行調優相信都會有一個很好的認知。 廢話不多說,直接帶大家來初步認識一下JVM。 什麼是JVM? JVM(Java Virtual Machine)是一個抽象的電腦,和實際的電腦一樣,它具有指令 ...


大家都知道,Java中JVM的重要性,學習了JVM你對Java的運行機制、編譯過程和如何對Java程式進行調優相信都會有一個很好的認知。

廢話不多說,直接帶大家來初步認識一下JVM。

  • 什麼是JVM?

JVM(Java Virtual Machine)是一個抽象的電腦,和實際的電腦一樣,它具有指令集並使用不同的存儲區域,它負責執行指令,還要管理數據、記憶體和寄存器。

看到這裡,可能不懂JVM的人,已經蒙圈了。沒關係,下麵讓我詳細為大家介紹JVM的體系架構圖,或許你會明白些。

簡單來說,JVM就是一個虛擬電腦。我們都知道Java語言其中的一個特性就是跨平臺的,而JVM就是Java程式實現跨平臺的關鍵部分。Java編譯器編譯Java程式時,生成的是與平臺無關的位元組碼(也就是*.class文件),所謂的平臺無關是指編譯生成的位元組碼無論是在Window、Linux、Mac系統都是可執行。也就是說Java編譯生成的*.class文件不是面向平臺的,而是面向JVM的。不同平臺上的JVM都是不同的,但是他們都是提供了相同的介面。圖一為Java的大致運行步驟:

 

 圖一

引用一個《瘋狂Java講義》中提到例子來幫助大家理解JVM的作用:

JVM的作用就像有兩隻不同的鉛筆,但需要把同一個筆帽套在兩支不同的筆上,只有為這兩支筆分別提供一個轉換器,這個轉換器向上的介面相同,用於適應同一個筆帽;向下的介面不同,用於適應兩支不同的筆。在這個類比中,可以近似地理解兩支不同的筆就是不同的操作系統,而同一個筆帽就是Java位元組碼程式,轉換器角色則對應JVM。類似地,也可以認為JVM分為向上和向下兩個部分,所有平臺的JVM向上提供給Java位元組碼程式的介面完全相同,但向下適應的不同平臺的介面則互不相同。

  • JVM體繫結構概覽

上面我們是初步介紹了JVM的作用,那麼要深入去瞭解JVM我們就需要瞭解JVM的體繫結構,請看圖二:

圖二

圖二是JVM的體系架構圖,接下讓我們一起來聊一聊每一個部分都是什麼意思。

1.類裝載器子系統(ClassLoader)

負責載入class文件,class文件在文件開頭有特定的文件標示,將class文件位元組碼內容載入到記憶體中,並將這些內容轉換成方法區中的運行時數據結構並且ClassLoader只負責class文件的載入,至於它是否可以運行,則由Execution Engine決定。

Java編譯生成的*.class文件就是通過ClassLoader進行載入的,那麼這裡就會有幾個問題:

  • ClassLoader如何知道*.class文件就是需要載入的文件?
  • 如果我手動將一個普通文件的擴展名稱改為class尾碼,ClassLoader會載入這個文件嗎?

實際上,class文件在文件的開頭是有特定的文件標識的,隨便編寫一個Java程式,編譯生成一個class文件,打開後你都能看到如下內容:

cafe babe就是class文件的一個標識,ClassLoader負責載入有cafe babe的class文件,它將class文件位元組碼內容載入到記憶體中,並將這些內容轉換成方法區中的運行時的數據結構並且ClassLoader只負責class文件的載入,至於它是否可以運行,則由Execution Engine決定,請看圖三:

 

 

圖三

Car.class文件通過ClassLoader進行載入到記憶體中,Car Class在記憶體中就相當一個模板,我們可以通過這個模板可以實例化成不同的實例car1、car2、car3。

不知大家會不會有一個疑問,ClassLoader載入Car.class在Java中是用什麼類型的載入器載入的呢?在解答這個問題前我們先寫個簡單的代碼看看:

        //new一個Car對象
        Car car = new Car();

        //得到ClassLoader
        ClassLoader classLoader = car.getClass().getClassLoader();

        //列印結果
        System.out.println(classLoader);

結果為:

 我們再來看看另外一組代碼:

        //new兩個不同的對象
        Car car = new Car();
        String string = new String();

        //得到ClassLoader
        ClassLoader classLoader1 = car.getClass().getClassLoader();
        ClassLoader classLoader2 = string.getClass().getClassLoader();

        //列印結果
        System.out.println(classLoader1);
        System.out.println(classLoader2);

結果為:

 從上面我們可以知道,ClassLoader的列印結果一個是“sun.misc.Launcher$AppClassLoader@18b4aac2”,一個則是“null”,這是怎麼回事呢,細心的朋友就可以發現這兩個不同的對象中,其中car對象是我們自己寫的一個類,string對象是系統自帶的一個類。簡單來說就是ClassLoader會根據不同的類選擇不同的類載入器去進行載入。這裡就牽扯到了ClassLoader的分類

ClassLoader的類別:

  • 啟動類載入器(BootStrap)
  • 擴展類載入器(Extension)
  • 應用程式類載入器(AppClassLoader)
  • 用戶自定義載入器

一般我們自己所寫的類用的類載入器都是AppClassLoader,就是上圖所示的“sun.misc.Launcher$AppClassLoader@18b4aac2”,而為什麼string這個對象是”null“呢?實際上,這個“null”指的就是使用BootStrap這個載入器。

那可能有人有疑問,自己定義的類用AppClassLoader,能理解,因為car這個對象輸出的類載入器名字中有AppClassLoader這個字樣,但是為什麼string這個對象是”null“,從哪裡可用體現是用BootStrap這個載入器呢?是這樣的,BootStrap累載入器相當於擴展類載入器、應用程式類載入器的祖宗,若是用了BootStrap,由於BootStrap上一級已經沒有了,所以就用“null”來表示

其實我們可以找一下String這個類在JDK的位置:

$JAVA_HOME/jre/lib/rt.jar/java/lang

所有在這個路徑$JAVA_HOME/jre/lib/rt.jar這個jar包下的類都是用BootStrap來載入的。

下麵請看圖4:

 

 

圖四 

這張圖就可以很清晰得看到:

1.所有在$Java_Home/jre/lib/rt.jar是通過BootStrap載入的

2.所有在$Java_Home/jre/lib/ext/*.jar是通過Extension載入的

3.所有在$CLASSPATH是通過SYSTEM載入的(應用程式類載入器也叫系統類載入器,載入當前應用的classpath的所有類)

接下來我們再來看一個例子:

如果創建一個java.lang包,然後創建String類,列印一句話執行會怎麼樣呢?

package java.lang;

public class String {
    public static void main(String[] args) {
        System.out.println("Hello World");
    }
}

效果如下:

 可以看到程式報錯了,說是找不到main方法,可是明明就有main方法為什麼沒有執行呢?這裡就涉及了雙親委派機制

雙親委派機制:

當一個類收到了類載入請求,他首先不會嘗試自己去載入這個類,而是把這個請求委派給父類去完成,每一個層次類載入器都是如此,因此所有的載入請求都應該傳送到啟動類載入器中,只有當父類載入器反饋自己無法完成這個請求的時候(在它的載入路徑下沒有找到所需載入的Class),子類載入器才會嘗試自己去載入。

所以它實際的運行過程是這樣的:

  • ClassLoader收到String類的載入請求。
  • 先去Bootstrap查找是否有這個類,沒有則反饋無法完成這個請求,但是恰好,在rt.jar中找到了java.lang.Stirng這個類
  • 執行這個類,這個類是沒有定義main方法的
  • 報錯,類中沒有定義main方法

所以上面的例子,他會找到jdk中java.lang.String這個類,這個類確實是沒有定義main方法,簡單來說它執行的類是JDK中java.lang.String這個類,而不是我們自己定義的類。

那用雙親委派機制有什麼好處呢:

採用雙親委派的一個好處是比如載入位於 rt.jar 包中的類 java.lang.Object,不管是哪個載入器載入這個類,最終都是委托給頂層的啟動類載入器進行載入,這樣就保證了使用不同的類載入器最終得到的都是同樣一個 Object對象。

2.執行引擎(Execution Engine)

執行引擎負責解釋命令,提交給操作系統執行,這裡對執行引擎就不做過多的解釋了,只要知道他是負責解釋命令的即可。

3.本地方法介面(Native Interface)和本地方法棧(Native Method Stack)

  • 本地介面:本地介面的作用是融合不同的編程語言為 Java 所用,它的初衷是融合 C/C++程式,Java 誕生的時候是 C/C++橫行的時候,要想立足,必須有調用 C/C++程式,於是就在記憶體中專門開闢了一塊區域處理標記為native的代碼,它的具體做法是 Native Method Stack中登記 native方法,在Execution Engine 執行時載入native libraies。

   目前該方法使用的越來越少了,除非是與硬體有關的應用,比如通過Java程式驅動印表機或者Java系統管理生產設備,在企業級應用中已經比較少見。因為現在的異構領域間的通信很發達,比如可以使用    Socket通信,也可以使用Web Service等等,不多做介紹。

          如果在程式中有見到native關鍵字,就代表不是Java能完成的事情了,需要載入本地方法庫才能完成

  • 本地方法棧:它的具體做法是Native Method Stack中登記native方法,在Execution Engine 執行時載入本地方法庫。說白了就是本地方法由本地方法棧來登記,Java中的方法由Java棧來登記。

4.PC寄存器(Program Counter Register)

每個線程都有一個程式計數器,是線程私有的,就是一個指針,指向方法區中的方法位元組碼(用來存儲指向下一條指令的地址,也即將要執行的指令代碼),由執行引擎讀取下一條指令,是一個非常小的記憶體空間,幾乎可以忽略不記。
這塊記憶體區域很小,它是當前線程所執行的位元組碼的行號指示器,位元組碼解釋器通過改變這個計數器的值來選取下一條需要執行的位元組碼指令。
如果執行的是一個Native方法,那這個計數器是空的。
PC寄存器用來完成分支、迴圈、跳轉、異常處理、線程恢復等基礎功能。由於使用的記憶體較小,所以不會發生記憶體溢出(OutOfMemory)錯誤。

那麼這篇文章先講到這裡,下篇文章中我們再繼續來聊一聊方法區、棧和堆..........


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

-Advertisement-
Play Games
更多相關文章
  • 中介者模式 定義 用一個中介對象來封裝一系列的對象交互。中介使各對象之間不需要顯示的相互引用,從而使其耦合鬆散,而且可以獨立的改變它們之間的交互。 UML圖 舉個慄子 “聯合國”就是世界上各個國家的一個“中介”,許多事情都不是國家之間直接交互,而是通過“安理會”等組織進行協商、投票等過程。 Talk ...
  • Java的HashMap源碼中用到的(n-1)&hash這樣的運算,查找發現這是一種高效的求餘數的辦法,但其中的原理是什麼呢為什麼可以這麼做呢? 先上結論:假設被除數是x,對於除數是2n的取餘操作x%2n,都可以寫成x&(2n-1),位運算效率高! eg:259%8=259&7=3 259 1000 ...
  • 參考文檔https://jingyan.baidu.com/article/4e5b3e190f55c591901e24b3.html admin.py from .models import *class BookAdmin(admin.ModelAdmin): list_display = [" ...
  • Java編譯時常量和運行時常量 編譯期常量指的就是程式在編譯時就能確定這個常量的具體值。 非編譯期常量就是程式在運行時才能確定常量的值,因此也稱為運行時常量。 在Java中,編譯期常量指的是用final關鍵字修飾的基本類型或String類型並直接賦值(非複雜運算)的變數(無論是否用static修飾) ...
  • nats的消息傳遞模型 @[toc] What is NATS nats是一個go語言開發的開源的、輕量、高性能的原生消息系統。nats消息由主題處理,不依賴於網路位置。它提供了應用程式或服務與底層物理網路之間的抽象層。數據被編碼並作為消息,由發佈者發送。消息由一個或多個訂閱者接收、解碼和處理。 N ...
  • 綜述 TensorFlow程式分為構建階段和執行階段。通過構建一個圖、執行這個圖來得到結果。 構建圖 創建源op,源op不需要任何輸入,例如常量 ,源op的輸出被傳遞給其他op做運算。 在一個會話中啟動圖 構造圖之後,需要創建 對象來啟動圖。 使用“with”代碼塊來自動完成關閉動作。 Tensor ...
  • 實現一個基於web技術的電影票訂票網站,故而系統主要以j2EE作為開發基礎,主要使用了struts2+spring+hibernate等多種框架的結合使用,用myeclipse作為開發工具,以MYSQL作為資料庫,以Macromedia公司的Dreamweaver作為界面美化工具,使用JAVA語言開 ...
  • 寫在前面 在我初次接觸MongoDB的時候,是為了做一個監控系統和日誌分析系統。當時在用Java操作MongoDB數據里的數據的時候,都是在網上查找Demo示例然後完成的功能,相信大家也同樣的體會,網上大都是這種一個入門的小Demo,很少有深入講解的。功能調通後,自己琢磨效率方面的問題,便開始自己的 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...