成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

深入理解Java虛擬機(jī)到底是什么

宋華 / 1317人閱讀

摘要:由虛擬機(jī)加載的類,被加載到虛擬機(jī)內(nèi)存中之后,虛擬機(jī)會(huì)讀取并執(zhí)行它里面存在的字節(jié)碼指令。虛擬機(jī)中執(zhí)行字節(jié)碼指令的部分叫做執(zhí)行引擎。

什么是Java虛擬機(jī)?

作為一個(gè)Java程序員,我們每天都在寫Java代碼,我們寫的代碼都是在一個(gè)叫做Java虛擬機(jī)的東西上執(zhí)行的。但是如果要問(wèn)什么是虛擬機(jī),恐怕很多人就會(huì)模棱兩可了。在本文中,我會(huì)寫下我對(duì)虛擬機(jī)的理解。因?yàn)槟芰λ蓿赡苡行┑胤矫枋龅牟粔蚯樊?dāng)。如果你有不同的理解,歡迎交流。

我們都知道java程序必須在虛擬機(jī)上運(yùn)行。那么虛擬機(jī)到底是什么呢?先看網(wǎng)上搜索到的比較靠譜的解釋:

虛擬機(jī)是一種抽象化的計(jì)算機(jī),通過(guò)在實(shí)際的計(jì)算機(jī)上仿真模擬各種計(jì)算機(jī)功能來(lái)實(shí)現(xiàn)的。Java虛擬機(jī)有自己完善的硬體架構(gòu),如處理器、堆棧、寄存器等,還具有相應(yīng)的指令系統(tǒng)。JVM屏蔽了與具體操作系統(tǒng)平臺(tái)相關(guān)的信息,使得Java程序只需生成在Java虛擬機(jī)上運(yùn)行的目標(biāo)代碼(字節(jié)碼),就可以在多種平臺(tái)上不加修改地運(yùn)行。

這種解釋應(yīng)該算是正確的,但是只描述了虛擬機(jī)的外部行為和功能,并沒有針對(duì)內(nèi)部原理做出說(shuō)明。一般情況下我們不需要知道虛擬機(jī)的運(yùn)行原理,只要專注寫java代碼就可以了,這也正是虛擬機(jī)之所以存在的原因--屏蔽底層操作系統(tǒng)平臺(tái)的不同并且減少基于原生語(yǔ)言開發(fā)的復(fù)雜性,使java這門語(yǔ)言能夠跨各種平臺(tái)(只要虛擬機(jī)廠商在特定平臺(tái)上實(shí)現(xiàn)了虛擬機(jī)),并且簡(jiǎn)單易用。這些都是虛擬機(jī)的外部特性,但是從這些信息來(lái)解釋虛擬機(jī),未免太籠統(tǒng)了,無(wú)法讓我們知道內(nèi)部原理。

從進(jìn)程的角度解釋JVM

讓我們嘗試從操作系統(tǒng)的層面來(lái)理解虛擬機(jī)。我們知道,虛擬機(jī)是運(yùn)行在操作系統(tǒng)之中的,那么什么東西才能在操作系統(tǒng)中運(yùn)行呢?當(dāng)然是進(jìn)程,因?yàn)檫M(jìn)程是操作系統(tǒng)中的執(zhí)行單位。可以這樣理解,當(dāng)它在運(yùn)行的時(shí)候,它就是一個(gè)操作系統(tǒng)中的進(jìn)程實(shí)例,當(dāng)它沒有在運(yùn)行時(shí)(作為可執(zhí)行文件存放于文件系統(tǒng)中),可以把它叫做程序。

對(duì)命令行比較熟悉的同學(xué),都知道其實(shí)一個(gè)命令對(duì)應(yīng)一個(gè)可執(zhí)行的二進(jìn)制文件,當(dāng)敲下這個(gè)命令并且回車后,就會(huì)創(chuàng)建一個(gè)進(jìn)程,加載對(duì)應(yīng)的可執(zhí)行文件到進(jìn)程的地址空間中,并且執(zhí)行其中的指令。下面對(duì)比C語(yǔ)言和Java語(yǔ)言的HelloWorld程序來(lái)說(shuō)明問(wèn)題。

首先編寫C語(yǔ)言版的HelloWorld程序。

#include   
#include   
 
int main(void) {  
   printf("hello world
");  
   return 0;  
}

編譯C語(yǔ)言版的HelloWorld程序:

gcc HelloWorld.c -o HelloWorld

運(yùn)行C語(yǔ)言版的HelloWorld程序:

zhangjg@linux:/deve/workspace/HelloWorld/src$ ./HelloWorld   
hello world

gcc編譯器編譯后的文件直接就是可被操作系統(tǒng)識(shí)別的二進(jìn)制可執(zhí)行文件,當(dāng)我們?cè)诿钚兄星孟?./HelloWorld這條命令的時(shí)候, 直接創(chuàng)建一個(gè)進(jìn)程, 并且將可執(zhí)行文件加載到進(jìn)程的地址空間中, 執(zhí)行文件中的指令。

作為對(duì)比, 我們看一下Java版HelloWord程序的編譯和執(zhí)行形式。

首先編寫源文件HelloWord.java :

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

編譯Java版的HelloWorld程序:

zhangjg@linux:/deve/workspace/HelloJava/src$ javac HelloWorld.java   
zhangjg@linux:/deve/workspace/HelloJava/src$ ls  
HelloWorld.class  HelloWorld.java

運(yùn)行Java版的HelloWorld程序:

zhangjg@linux:/deve/workspace/HelloJava/src$ java -classpath . HelloWorld   
HelloWorld

從上面的過(guò)程可以看到, 我們?cè)谶\(yùn)行Java版的HelloWorld程序的時(shí)候, 敲入的命令并不是 ./HelloWorld.class 。 因?yàn)閏lass文件并不是可以直接被操作系統(tǒng)識(shí)別的二進(jìn)制可執(zhí)行文件 。 我們敲入的是java這個(gè)命令。 這個(gè)命令說(shuō)明, 我們首先啟動(dòng)的是一個(gè)叫做java的程序, 這個(gè)java程序在運(yùn)行起來(lái)之后就是一個(gè)JVM進(jìn)程實(shí)例。

上面的命令執(zhí)行流程是這樣的:

java命令首先啟動(dòng)虛擬機(jī)進(jìn)程,虛擬機(jī)進(jìn)程成功啟動(dòng)后,讀取參數(shù)“HelloWorld”,把他作為初始類加載到內(nèi)存,對(duì)這個(gè)類進(jìn)行初始化和動(dòng)態(tài)鏈接(關(guān)于類的初始化和動(dòng)態(tài)鏈接會(huì)在后面的文章中介紹),然后從這個(gè)類的main方法開始執(zhí)行。也就是說(shuō)我們的.class文件不是直接被系統(tǒng)加載后直接在cpu上執(zhí)行的,而是被一個(gè)叫做虛擬機(jī)的進(jìn)程托管的。首先必須虛擬機(jī)進(jìn)程啟動(dòng)就緒,然后由虛擬機(jī)中的類加載器加載必要的class文件,包括jdk中的基礎(chǔ)類(如String和Object等),然后由虛擬機(jī)進(jìn)程解釋class字節(jié)碼指令,把這些字節(jié)碼指令翻譯成本機(jī)cpu能夠識(shí)別的指令,才能在cpu上運(yùn)行。

從這個(gè)層面上來(lái)看,在執(zhí)行一個(gè)所謂的java程序的時(shí)候,真真正正在執(zhí)行的是一個(gè)叫做Java虛擬機(jī)的進(jìn)程,而不是我們寫的一個(gè)個(gè)的class文件。這個(gè)叫做虛擬機(jī)的進(jìn)程處理一些底層的操作,比如內(nèi)存的分配和釋放等等。我們編寫的class文件只是虛擬機(jī)進(jìn)程執(zhí)行時(shí)需要的“原料”。這些“原料”在運(yùn)行時(shí)被加載到虛擬機(jī)中,被虛擬機(jī)解釋執(zhí)行,以控制虛擬機(jī)實(shí)現(xiàn)我們java代碼中所定義的一些相對(duì)高層的操作,比如創(chuàng)建一個(gè)文件等,可以將class文件中的信息看做對(duì)虛擬機(jī)的控制信息,也就是一種虛擬指令。

編程語(yǔ)言也有自己的原理, 學(xué)習(xí)一門語(yǔ)言, 主要是把它的原理搞明白。 看似一個(gè)簡(jiǎn)單的HelloWorld程序, 也有很多深入的內(nèi)容值得剖析。

JVM體系結(jié)構(gòu)簡(jiǎn)介

為了展示虛擬機(jī)進(jìn)程和class文件的關(guān)系,特意畫了下面一張圖:

根據(jù)上圖表達(dá)的內(nèi)容,我們編譯之后的class文件是作為Java虛擬機(jī)的原料被輸入到Java虛擬機(jī)的內(nèi)部的,那么具體由誰(shuí)來(lái)做這一部分工作呢?其實(shí)在Java虛擬機(jī)內(nèi)部,有一個(gè)叫做類加載器的子系統(tǒng),這個(gè)子系統(tǒng)用來(lái)在運(yùn)行時(shí)根據(jù)需要加載類。注意上面一句話中的“根據(jù)需要”四個(gè)字。在Java虛擬機(jī)執(zhí)行過(guò)程中,只有他需要一個(gè)類的時(shí)候,才會(huì)調(diào)用類加載器來(lái)加載這個(gè)類,并不會(huì)在開始運(yùn)行時(shí)加載所有的類。就像一個(gè)人,只有餓的時(shí)候才去吃飯,而不是一次把一年的飯都吃到肚子里。一般來(lái)說(shuō),虛擬機(jī)加載類的時(shí)機(jī),在第一次使用一個(gè)新的類的時(shí)候。后面的文章會(huì)具體討論Java中的類加載器。

由虛擬機(jī)加載的類,被加載到Java虛擬機(jī)內(nèi)存中之后,虛擬機(jī)會(huì)讀取并執(zhí)行它里面存在的字節(jié)碼指令。虛擬機(jī)中執(zhí)行字節(jié)碼指令的部分叫做執(zhí)行引擎。就像一個(gè)人,不是把飯吃下去就完事了,還要進(jìn)行消化,執(zhí)行引擎就相當(dāng)于人的腸胃系統(tǒng)。在執(zhí)行的過(guò)程中還會(huì)把各個(gè)class文件動(dòng)態(tài)的連接起來(lái)。關(guān)于執(zhí)行引擎的具體行為和動(dòng)態(tài)鏈接相關(guān)的內(nèi)容也會(huì)在后續(xù)的文章中進(jìn)行討論。

我們知道,Java虛擬機(jī)會(huì)進(jìn)行自動(dòng)內(nèi)存管理。具體說(shuō)來(lái)就是自動(dòng)釋放沒有用的對(duì)象,而不需要程序員編寫代碼來(lái)釋放分配的內(nèi)存。這部分工作由垃圾收集子系統(tǒng)負(fù)責(zé)。

從上面的論述可以知道, 一個(gè)Java虛擬機(jī)實(shí)例在運(yùn)行過(guò)程中有三個(gè)子系統(tǒng)來(lái)保障它的正常運(yùn)行,分別是類加載器子系統(tǒng), 執(zhí)行引擎子系統(tǒng)和垃圾收集子系統(tǒng)。 如下圖所示:

虛擬機(jī)的運(yùn)行,必須加載class文件,并且執(zhí)行class文件中的字節(jié)碼指令。它做這么多事情,必須需要自己的空間。就像人吃下去的東西首先要放在胃中。虛擬機(jī)也需要空間來(lái)存放個(gè)中數(shù)據(jù)。首先,加載的字節(jié)碼,需要一個(gè)多帶帶的內(nèi)存空間來(lái)存放;一個(gè)線程的執(zhí)行,也需要內(nèi)存空間來(lái)維護(hù)方法的調(diào)用關(guān)系,存放方法中的數(shù)據(jù)和中間計(jì)算結(jié)果;在執(zhí)行的過(guò)程中,無(wú)法避免的要?jiǎng)?chuàng)建對(duì)象,創(chuàng)建的對(duì)象需要一個(gè)專門的內(nèi)存空間來(lái)存放。關(guān)于虛擬機(jī)運(yùn)行時(shí)數(shù)據(jù)區(qū)的內(nèi)容,也會(huì)出現(xiàn)在后續(xù)的文章中。虛擬機(jī)的運(yùn)行時(shí)內(nèi)存區(qū)大概可以分成下圖所示的幾個(gè)部分。(這里只是大概劃分, 并沒有劃分的很精細(xì))

總結(jié)

寫到這里,基本上關(guān)于我對(duì)java虛擬機(jī)的理解就寫完了。這篇文章的主題雖然是深入理解Java虛擬機(jī),但是你可能感覺一點(diǎn)也不“深入”,也只是泛泛而談。我也有這樣的感覺。限于自己水平有限,也只能這樣了,要是想深入理解java虛擬機(jī),強(qiáng)烈建議讀一下三本書:

《深入Java虛擬機(jī)》

《深入理解Java虛擬機(jī)JVM高級(jí)特性與最佳實(shí)踐》

《Java虛擬機(jī)規(guī)范》

其實(shí)我也讀過(guò)這幾本書,但是它們對(duì)虛擬機(jī)的解釋也是基于一個(gè)外部模型,而沒有深入剖析虛擬機(jī)內(nèi)部的實(shí)現(xiàn)原理。虛擬機(jī)是一個(gè)大而復(fù)雜的東西,實(shí)現(xiàn)虛擬機(jī)的人都是大牛級(jí)別的,如果不是參與過(guò)虛擬機(jī)的實(shí)現(xiàn),應(yīng)該很少有人能把它參透。后面的一些文章也參考了這三本書, 雖然講解Java語(yǔ)法的書不計(jì)其數(shù), 但是深入講解虛擬機(jī)的書, 目前為止我就見過(guò)這三本,并且網(wǎng)上的資料也不是很多。

最后做一個(gè)總結(jié):

虛擬機(jī)并不神秘,在操作系統(tǒng)的角度看來(lái),它只是一個(gè)普通進(jìn)程。

這個(gè)叫做虛擬機(jī)的進(jìn)程比較特殊,它能夠加載我們編寫的class文件。如果把JVM比作一個(gè)人,那么class文件就是我們吃的食物。

加載class文件的是一個(gè)叫做類加載器的子系統(tǒng)。就好比我們的嘴巴,把食物吃到肚子里。

虛擬機(jī)中的執(zhí)行引擎用來(lái)執(zhí)行class文件中的字節(jié)碼指令。就好比我們的腸胃,對(duì)吃進(jìn)去的食物進(jìn)行消化。

虛擬機(jī)在執(zhí)行過(guò)程中,要分配內(nèi)存創(chuàng)建對(duì)象。當(dāng)這些對(duì)象過(guò)時(shí)無(wú)用了,必須要自動(dòng)清理這些無(wú)用的對(duì)象。清理對(duì)象回收內(nèi)存的任務(wù)由垃圾收集器負(fù)責(zé)。就好比人吃進(jìn)去的食物,在消化之后,必須把廢物排出體外,騰出空間可以在下次餓的時(shí)候吃飯并消化食物。

我有一個(gè)微信公眾號(hào),經(jīng)常會(huì)分享一些Java技術(shù)相關(guān)的干貨;如果你喜歡我的分享,可以用微信搜索“Java團(tuán)長(zhǎng)”或者“javatuanzhang”關(guān)注。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/69204.html

相關(guān)文章

  • 從表到里學(xué)習(xí)JVM實(shí)現(xiàn)

    在社會(huì)化分工、軟件行業(yè)細(xì)分專業(yè)化的趨勢(shì)下,會(huì)真的參與到底層系統(tǒng)實(shí)現(xiàn)的人肯定是越來(lái)越少(比例上說(shuō))。真的會(huì)參與到JVM實(shí)現(xiàn)的人肯定是少數(shù)。 但如果您對(duì)JVM是如何實(shí)現(xiàn)的有興趣、充滿好奇,卻苦于沒有足夠系統(tǒng)的知識(shí)去深入,那么可以參考RednaxelaFX整理的這個(gè)書單。 showImg(http://segmentfault.com/img/bVbGzn); 本豆列的脈絡(luò)是:    1. JV...

    Cristic 評(píng)論0 收藏0
  • 深入理解虛擬機(jī)虛擬機(jī)性能監(jiān)控和故障處理工具

    摘要:監(jiān)控和故障處理工具顯示指定系統(tǒng)內(nèi)所有的虛擬機(jī)進(jìn)程用于收集虛擬機(jī)各方面的運(yùn)行數(shù)據(jù)。的常用功能選項(xiàng)測(cè)試上面輸出了我正在運(yùn)行程序的包名下的類名虛擬機(jī)統(tǒng)計(jì)信息監(jiān)視工具使用于監(jiān)視虛擬機(jī)各種運(yùn)行狀態(tài)信息的命令行工具。 《深入理解Java虛擬機(jī):JVM高級(jí)特性與最佳實(shí)踐(第二版》讀書筆記與常見面試題總結(jié) 本節(jié)常見面試題(推薦帶著問(wèn)題閱讀,問(wèn)題答案在文中都有提到): JVM調(diào)優(yōu)的常見命令行工具有哪些?...

    jas0n 評(píng)論0 收藏0
  • Java虛擬機(jī)中,字符串常量到底存放在哪

    摘要:的三種常量池此外,有三種常量池,即字符串常量池又叫全局字符串池文件常量池運(yùn)行時(shí)常量池。開始虛擬機(jī)把字符串常量池位置從永久代挪到堆,又徹底取消,把諸如之類的元數(shù)據(jù)都挪到堆之外管理。 前言 前陣子和朋友討論一個(gè)問(wèn)題: 字符串常量歸常量池管理,那比如 String str = abc; abc這個(gè)對(duì)象是放在內(nèi)存中的哪個(gè)位置,是字符串常量池中還是堆? 這句代碼的abc當(dāng)然在常量池中,只有n...

    lewinlee 評(píng)論0 收藏0
  • java 基礎(chǔ) - 收藏集 - 掘金

    摘要:基礎(chǔ)知識(shí)復(fù)習(xí)后端掘金的作用表示靜態(tài)修飾符,使用修飾的變量,在中分配內(nèi)存后一直存在,直到程序退出才釋放空間。將對(duì)象編碼為字節(jié)流稱之為序列化,反之將字節(jié)流重建成對(duì)象稱之為反序列化。 Java 學(xué)習(xí)過(guò)程|完整思維導(dǎo)圖 - 后端 - 掘金JVM 1. 內(nèi)存模型( 內(nèi)存分為幾部分? 堆溢出、棧溢出原因及實(shí)例?線上如何排查?) 2. 類加載機(jī)制 3. 垃圾回收 Java基礎(chǔ) 什么是接口?什么是抽象...

    makeFoxPlay 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<