JVM工作原理和特點(diǎn)探究
我們運(yùn)行和調(diào)試Java程序的時(shí)候,經(jīng)常會(huì)提到一個(gè)JVM的概念,這里向大家描述一下JVM工作原理和特點(diǎn),Java虛擬機(jī)(JVM)一種用于計(jì)算設(shè)備的規(guī)范,可用不同的方式(軟件或硬件)加以實(shí)現(xiàn)。
JVM工作原理和特點(diǎn)
我們運(yùn)行和調(diào)試Java程序的時(shí)候,經(jīng)常會(huì)提到一個(gè)JVM的概念.JVM是Java程序運(yùn)行的環(huán)境,但是他同時(shí)一個(gè)操作系統(tǒng)的一個(gè)應(yīng)用程序一個(gè)進(jìn)程,因此他也有他自己的運(yùn)行的生命周期,也有自己的代碼和數(shù)據(jù)空間.
首先來說一下JVM工作原理中的jdk這個(gè)東西,不管你是初學(xué)者還是高手,是j2ee程序員還是j2se程序員,jdk總是在幫我們做一些事情.我們?cè)诹私釰ava之前首先大師們會(huì)給我們提供說jdk這個(gè)東西.它在Java整個(gè)體系中充當(dāng)著什么角色呢?我很驚嘆sun大師們?cè)O(shè)計(jì)天才,能把一個(gè)如此完整的體系結(jié)構(gòu)化的如此***.jdk在這個(gè)體系中充當(dāng)一個(gè)生產(chǎn)加工中心,產(chǎn)生所有的數(shù)據(jù)輸出,是所有指令和戰(zhàn)略的執(zhí)行中心.本身它提供了Java的完整方案,可以開發(fā)目前Java能支持的所有應(yīng)用和系統(tǒng)程序.這里說一個(gè)問題,大家會(huì)問,那為什么還有j2me,j2ee這些東西,這兩個(gè)東西目的很簡單,分別用來簡化各自領(lǐng)域內(nèi)的開發(fā)和構(gòu)建過程.jdk除了JVM之外,還有一些核心的API,集成API,用戶工具,開發(fā)技術(shù),開發(fā)工具和API等組成
好了,廢話說了那么多,來點(diǎn)于主題相關(guān)的東西吧.JVM在整個(gè)jdk中處于***層,負(fù)責(zé)于操作系統(tǒng)的交互,用來屏蔽操作系統(tǒng)環(huán)境,提供一個(gè)完整的Java運(yùn)行環(huán)境,因此也就虛擬計(jì)算機(jī).操作系統(tǒng)裝入JVM是通過jdk中Java.exe來完成,通過下面4步來完成JVM環(huán)境.
1.創(chuàng)建JVM裝載環(huán)境和配置
2.裝載JVM.dll
3.初始化JVM.dll并掛界到JNIENV(JNI調(diào)用接口)實(shí)例
4.調(diào)用JNIEnv實(shí)例裝載并處理class類。
一.JVM裝入環(huán)境
JVM提供的方式是操作系統(tǒng)的動(dòng)態(tài)連接文件.既然是文件那就一個(gè)裝入路徑的問題,Java是怎么找這個(gè)路徑的呢?當(dāng)你在調(diào)用Javatest的時(shí)候,操作系統(tǒng)會(huì)在path下在你的Java.exe程序,Java.exe就通過下面一個(gè)過程來確定JVM的路徑和相關(guān)的參數(shù)配置了.下面基于Windows的實(shí)現(xiàn)的分析.
◆首先查找jre路徑,Java是通過GetApplicationHomeapi來獲得當(dāng)前的Java.exe絕對(duì)路徑,c:\j2sdk1.4.2_09\bin\Java.exe,那么它會(huì)截取到絕對(duì)路徑c:\j2sdk1.4.2_09\,判斷c:\j2sdk1.4.2_09\bin\Java.dll文件是否存在,如果存在就把c:\j2sdk1.4.2_09\作為jre路徑,如果不存在則判斷c:\j2sdk1.4.2_09\jre\bin\Java.dll是否存在,如果存在這c:\j2sdk1.4.2_09\jre作為jre路徑.如果不存在調(diào)用GetPublicJREHome查HKEY_LOCAL_MACHINE\Software\JavaSoft\JavaRuntimeEnvironment\“當(dāng)前JRE版本號(hào)”\JavaHome的路徑為jre路徑。
◆然后裝載JVM.cfg文件JRE路徑+\lib+\ARCH(CPU構(gòu)架)+\JVM.cfgARCH(CPU構(gòu)架)的判斷是通過Java_md.c中GetArch函數(shù)判斷的,該函數(shù)中windows平臺(tái)只有兩種情況:WIN64的‘ia64’,其他情況都為‘i386’。以我的為例:C:\j2sdk1.4.2_09\jre\lib\i386\JVM.cfg.主要的內(nèi)容如下:
-clientKNOWN-serverKNOWN-hotspotALIASED_TO-client-classicWARN-nativeERROR-greenERROR在我們的jdk目錄中jre\bin\server和jre\bin\client都有JVM.dll文件存在,而Java正是通過JVM.cfg配置文件來管理這些不同版本的JVM.dll的.通過文件我們可以定義目前jdk中支持那些JVM,前面部分(client)是JVM名稱,后面是參數(shù),KNOWN表示JVM存在,ALIASED_TO表示給別的JVM取一個(gè)別名,WARN表示不存在時(shí)找一個(gè)JVM替代,ERROR表示不存在拋出異常.
◆在運(yùn)行JavaXXX是,Java.exe會(huì)通過CheckJVMType來檢查當(dāng)前的JVM類型,Java可以通過兩種參數(shù)的方式來指定具體的JVM類型,一種按照J(rèn)VM.cfg文件中的JVM名稱指定,第二種方法是直接指定,它們執(zhí)行的方法分別是“Java-J”、“Java-XXaltJVM=”或“Java-J-XXaltJVM=”。
如果是***種參數(shù)傳遞方式,CheckJVMType函數(shù)會(huì)取參數(shù)‘-J’后面的JVM名稱,然后從已知的JVM配置參數(shù)中查找如果找到同名的則去掉該JVM名稱前的‘-’直接返回該值;而第二種方法,會(huì)直接返回“-XXaltJVM=”或“-J-XXaltJVM=”后面的JVM類型名稱;如果在運(yùn)行Java時(shí)未指定上面兩種方法中的任一一種參數(shù),CheckJVMType會(huì)取配置文件中***個(gè)配置中的JVM名稱,去掉名稱前面的‘-’返回該值。
CheckJVMType函數(shù)的這個(gè)返回值會(huì)在下面的函數(shù)中匯同jre路徑組合成JVM.dll的絕對(duì)路徑。如果沒有指定這會(huì)使用JVM.cfg中***個(gè)定義的JVM.可以通過set_Java_LAUNCHER_DEBUG=1在控制臺(tái)上測試.
***獲得JVM.dll的路徑,JRE路徑+\bin+\JVM類型字符串+\JVM.dll就是JVM的文件路徑了,但是如果在調(diào)用Java程序時(shí)用-XXaltJVM=參數(shù)指定的路徑path,就直接用path+\JVM.dll文件做為JVM.dll的文件路徑.#p#
二:裝載JVM.dll
通過***步已經(jīng)找到了JVM的路徑,Java通過LoadJavaVM來裝入JVM.dll文件.裝入工作很簡單就是調(diào)用WindowsAPI函數(shù):
LoadLibrary裝載JVM.dll動(dòng)態(tài)連接庫.然后把JVM.dll中的導(dǎo)出函數(shù)JNI_CreateJavaVM和JNI_GetDefaultJavaVMInitArgs掛接到InvocationFunctions變量的CreateJavaVM和GetDefaultJavaVMInitArgs函數(shù)指針變量上。JVM.dll的裝載工作宣告完成。
三:初始化JVM,獲得本地調(diào)用接口
這樣就可以在Java中調(diào)用JVM的函數(shù)了.調(diào)用InvocationFunctions->CreateJavaVM也就是JVM中JNI_CreateJavaVM方法獲得JNIEnv結(jié)構(gòu)的實(shí)例.
四:運(yùn)行Java程序.
Java程序有兩種方式一種是jar包,一種是class.運(yùn)行jar,Java-jarXXX.jar運(yùn)行的時(shí)候,Java.exe調(diào)用GetMainClassName函數(shù),該函數(shù)先獲得JNIEnv實(shí)例然后調(diào)用Java類Java.util.jar.JarFileJNIEnv中方法getManifest()并從返回的Manifest對(duì)象中取getAttributes("Main-Class")的值即jar包中文件:META-INF/MANIFEST.MF指定的Main-Class的主類名作為運(yùn)行的主類。之后main函數(shù)會(huì)調(diào)用Java.c中LoadClass方法裝載該主類(使用JNIEnv實(shí)例的FindClass)。main函數(shù)直接調(diào)用Java.c中LoadClass方法裝載該類。如果是執(zhí)行class方法。main函數(shù)直接調(diào)用Java.c中LoadClass方法裝載該類。
然后main函數(shù)調(diào)用JNIEnv實(shí)例的GetStaticMethodID方法查找裝載的class主類中
“publicstaticvoidmain(String[]args)”方法,并判斷該方法是否為public方法,然后調(diào)用JNIEnv實(shí)例的
CallStaticVoidMethod方法調(diào)用該Java類的main方法。
JVM工作原理和特點(diǎn)介紹到這里。
【編輯推薦】
- Java虛擬機(jī)概念及體系結(jié)構(gòu)
- 發(fā)現(xiàn)Java虛擬機(jī)內(nèi)存泄露問題
- 深入Java虛擬機(jī):JVM中的Stack和Heap
- Java虛擬機(jī)發(fā)展回顧 為跨平臺(tái)而生
- Java虛擬機(jī)(JVM)中的內(nèi)存設(shè)置詳解