當前位置:編程學習大全網 - 編程語言 - 什麽是Java虛擬機?

什麽是Java虛擬機?

虛擬機是壹種抽象化的計算機,通過在實際的計算機上仿真模擬各種計算機功能來實現的。

Java虛擬機有自己完善的硬體架構,如處理器、堆棧、寄存器等,還具有相應的指令系統。JVM屏蔽了與具體操作系統平臺相關的信息,使得Java程序只需生成在Java虛擬機上運行的目標代碼(字節碼),就可以在多種平臺上不加修改地運行。

這種解釋應該算是正確的,但是只描述了虛擬機的外部行為和功能,並沒有針對內部原理做出說明。壹般情況下我們不需要知道虛擬機的運行原理,只要專註寫java代碼就可以了,這也正是虛擬機之所以存在的原因--屏蔽底層操作系統平臺的不同並且減少基於原生語言開發的復雜性,使java這門語言能夠跨各種平臺(只要虛擬機廠商在特定平臺上實現了虛擬機),並且簡單易用。這些都是虛擬機的外部特性,但是從這些信息來解釋虛擬機,未免太籠統了,無法讓我們知道內部原理。

從進程的角度解釋JVM

讓我們嘗試從操作系統的層面來理解虛擬機。我們知道,虛擬機是運行在操作系統之中的,那麽什麽東西才能在操作系統中運行呢?當然是進程,因為進程是操作系統中的執行單位。可以這樣理解,當它在運行的時候,它就是壹個操作系統中的進程實例,當它沒有在運行時(作為可執行文件存放於文件系統中),可以把它叫做程序。

對命令行比較熟悉的同學,都知道其實壹個命令對應壹個可執行的二進制文件,當敲下這個命令並且回車後,就會創建壹個進程,加載對應的可執行文件到進程的地址空間中,並且執行其中的指令。下面對比C語言和Java語言的HelloWorld程序來說明問題。

首先編寫C語言版的HelloWorld程序。

編譯C語言版的HelloWorld程序:

gcc HelloWorld.c -o HelloWorld

運行C語言版的HelloWorld程序:

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

hello world

gcc編譯器編譯後的文件直接就是可被操作系統識別的二進制可執行文件,當我們在命令行中敲下 ./HelloWorld這條命令的時候, 直接創建壹個進程, 並且將可執行文件加載到進程的地址空間中, 執行文件中的指令。

作為對比, 我們看壹下Java版HelloWord程序的編譯和執行形式。

首先編寫源文件HelloWord.java :

編譯Java版的HelloWorld程序:

運行Java版的HelloWorld程序:

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

HelloWorld

從上面的過程可以看到, 我們在運行Java版的HelloWorld程序的時候, 敲入的命令並不是 ./HelloWorld.class 。 因為class文件並不是可以直接被操作系統識別的二進制可執行文件 。 我們敲入的是java這個命令。 這個命令說明, 我們首先啟動的是壹個叫做java的程序, 這個java程序在運行起來之後就是壹個JVM進程實例。

上面的命令執行流程是這樣的:

java命令首先啟動虛擬機進程,虛擬機進程成功啟動後,讀取參數“HelloWorld”,把他作為初始類加載到內存,對這個類進行初始化和動態鏈接(關於類的初始化和動態鏈接會在後面的博客中介紹),然後從這個類的main方法開始執行。也就是說我們的.class文件不是直接被系統加載後直接在cpu上執行的,而是被壹個叫做虛擬機的進程托管的。首先必須虛擬機進程啟動就緒,然後由虛擬機中的類加載器加載必要的class文件,包括jdk中的基礎類(如String和Object等),然後由虛擬機進程解釋class字節碼指令,把這些字節碼指令翻譯成本機cpu能夠識別的指令,才能在cpu上運行。

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

編程語言也有自己的原理, 學習壹門語言, 主要是把它的原理搞明白。 看似壹個簡單的HelloWorld程序, 也有很多深入的內容值得剖析。

JVM體系結構簡介

為了展示虛擬機進程和class文件的關系,特意畫了下面壹張圖:

根據上圖表達的內容,我們編譯之後的class文件是作為Java虛擬機的原料被輸入到Java虛擬機的內部的,那麽具體由誰來做這壹部分工作呢?其實在Java虛擬機內部,有壹個叫做類加載器的子系統,這個子系統用來在運行時根據需要加載類。註意上面壹句話中的“根據需要”四個字。在Java虛擬機執行過程中,只有他需要壹個類的時候,才會調用類加載器來加載這個類,並不會在開始運行時加載所有的類。就像壹個人,只有餓的時候才去吃飯,而不是壹次把壹年的飯都吃到肚子裏。壹般來說,虛擬機加載類的時機,在第壹次使用壹個新的類的時候。本專欄後面的文章會具體討論Java中的類加載器。

由虛擬機加載的類,被加載到Java虛擬機內存中之後,虛擬機會讀取並執行它裏面存在的字節碼指令。虛擬機中執行字節碼指令的部分叫做執行引擎。就像壹個人,不是把飯吃下去就完事了,還要進行消化,執行引擎就相當於人的腸胃系統。在執行的過程中還會把各個class文件動態的連接起來。關於執行引擎的具體行為和動態鏈接相關的內容也會在本專欄後續的文章中進行討論。

我們知道,Java虛擬機會進行自動內存管理。具體說來就是自動釋放沒有用的對象,而不需要程序員編寫代碼來釋放分配的內存。這部分工作由垃圾收集子系統負責。

從上面的論述可以知道, 壹個Java虛擬機實例在運行過程中有三個子系統來保障它的正常運行,分別是類加載器子系統, 執行引擎子系統和垃圾收集子系統。 如下圖所示:

虛擬機的運行,必須加載class文件,並且執行class文件中的字節碼指令。它做這麽多事情,必須需要自己的空間。就像人吃下去的東西首先要放在胃中。虛擬機也需要空間來存放個中數據。首先,加載的字節碼,需要壹個單獨的內存空間來存放;壹個線程的執行,也需要內存空間來維護方法的調用關系,存放方法中的數據和中間計算結果;在執行的過程中,無法避免的要創建對象,創建的對象需要壹個專門的內存空間來存放。關於虛擬機運行時數據區的內容,也會出現在本專欄後續的文章中。虛擬機的運行時內存區大概可以分成下圖所示的幾個部分。(這裏只是大概劃分,並沒有劃分的很精細)

總結

寫到這裏,基本上關於我對java虛擬機的理解就寫完了。這篇文章的主題雖然是深入理解Java虛擬機,但是妳可能感覺壹點也不“深入”,也只是泛泛而談。我也有這樣的感覺。限於自己水平有限,也只能這樣了,要是想深入理解java虛擬機,強烈建議讀壹下三本書:

《深入Java虛擬機》

《深入理解Java虛擬機JVM高級特性與最佳實踐》

《Java虛擬機規範》

其實我也讀過這幾本書,但是它們對虛擬機的解釋也是基於壹個外部模型,而沒有深入剖析虛擬機內部的實現原理。虛擬機是壹個大而復雜的東西,實現虛擬機的人都是大牛級別的,如果不是參與過虛擬機的實現,應該很少有人能把它參透。本專欄後面的壹些文章也參考了這三本書, 雖然講解Java語法的書不計其數, 但是深入講解虛擬機的書, 目前為止我就見過這三本,並且網上的資料也不是很多。

最後做壹個總結:

1 虛擬機並不神秘,在操作系統的角度看來,它只是壹個普通進程。

2 這個叫做虛擬機的進程比較特殊,它能夠加載我們編寫的class文件。如果把JVM比作壹個人,那麽class文件就是我們吃的食物。

3 加載class文件的是壹個叫做類加載器的子系統。就好比我們的嘴巴,把食物吃到肚子裏。

4 虛擬機中的執行引擎用來執行class文件中的字節碼指令。就好比我們的腸胃,對吃進去的食物進行消化。

5 虛擬機在執行過程中,要分配內存創建對象。當這些對象過時無用了,必須要自動清理這些無用的對象。清理對象回收內存的任務由垃圾收集器負責。就好比人吃進去的食物,在消化之後,必須把廢物排出體外,騰出空間可以在下次餓的時候吃飯並消化食物。

擴展資料:

關於JAVA虛擬機的參數說明如下:

1、運行class文件

執行帶main方法的class文件,Java虛擬機[3]?命令參數行為:

java <CLASS文件名>

註意:CLASS文件名不要帶文件後綴.class

例如:

java Test

如果執行的class文件是帶包的,即在類文件中使用了:

package <;包名>

那應該在包的基路徑下執行,Java虛擬機命令行參數:

java <;包名>.CLASS文件名

例如:

PackageTest.java中,其包名為:com.ee2ee.test,對應的語句為:

package com.ee2ee.test;

PackageTest.java及編譯後的class文件PackageTest.class的存放目錄如下:

classes

|__com

|__ee2ee

|__test

|__PackageTest.java

|__PackageTest.class

要運行PackageTest.class,應在classes目錄下執行:

java com.ee2ee.test.PackageTest

2、運行jar文件中的class

原理和運行class文件壹樣,只需加上參數-cp <jar文件名>;即可。

例如:執行test.jar中的類com.ee2ee.test.PackageTest,命令行如下:

java -cp test.jar com.ee2ee.test.PackageTest

3、顯示JDK版本信息

當壹臺機器上有多個jdk版本時,需要知道當前使用的是那個版本的jdk,使用參數-version即可知道其版本,命令行為:

java -version

4、增加虛擬機可以使用的最大內存

Java虛擬機可使用的最大內存是有限制的,缺省值通常為64MB或128MB。

如果壹個應用程序為了提高性能而把數據加載內存中而占用較大的內存,比如超過了默認的最大值128MB,需要加大java虛擬機可使用的最大內存,否則會出現Out of Memory的異常。啟動java時,需要使用如下兩個參數:

-Xms java虛擬機初始化時使用的內存大小

-Xmx java虛擬機可以使用的最大內存

以上兩個命令行參數中設置的size,可以帶單位,例如:256m表示256MB

舉例說明:

java -Xms128m -Xmx256m ...

表示Java虛擬機初始化時使用的內存為128MB,可使用的最大內存為256MB。

對於tomcat,可以修改其腳本catalina. sh(Unix平臺)或catalina.bat(Windows平臺),設置變量JAVA_OPTS即可,例如:

JAVA_OPTS='-Xms128m -Xmx256m'

參考資料:

百度百科-java虛擬機

  • 上一篇:python 和 r 的區別 知乎
  • 下一篇:AT命令的AT命令集
  • copyright 2024編程學習大全網