一、一個命令對應(yīng)一個進(jìn)程。
當(dāng)我們啟動一個Java程序,即啟動一個main方法時,都將啟動一個Java虛擬機(jī)進(jìn)程,不管這個進(jìn)程有多么復(fù)雜。而不同的JVM進(jìn)程之間是不會相互影響的。這也就是為什么說,Java程序只有一個入口——main方法,讓虛擬機(jī)調(diào)用。而兩個mian方法,對應(yīng)的是2個JVM進(jìn)程,啟動的是兩個不同的類加載器,操作的實(shí)際上是不同的類。故而不會互相影響。
二、類加載。
當(dāng)我們使用一個類,如果這個類還未加載到內(nèi)存中,系統(tǒng)會通過加載、連接、初始化對類進(jìn)行初始化。
1、類加載:指的是將類的class文件讀入JVM,并為之創(chuàng)建一個Class對象。
2、類連接:指的是把類的二進(jìn)制數(shù)據(jù)合并到JRE中,這又分為3個階段:
a)、校驗(yàn):檢查載入Class文件數(shù)據(jù)的正確性。
b)、準(zhǔn)備:給類的靜態(tài)變量分配存儲空間,并進(jìn)行默認(rèn)初始化。
c)、解析:將類的二進(jìn)制數(shù)據(jù)中的符號引用替換成直接引用。
3、初始化:對類的靜態(tài)變量、靜態(tài)初始化塊進(jìn)行初始化。
(注意:一個final類型的靜態(tài)屬性,如果在編譯時已經(jīng)得到了屬性值,那么調(diào)用該屬性時,不會導(dǎo)致該類初始化,因?yàn)檫@個相當(dāng)于使用常量;
使用ClassLoader()方法,只是加載該類,并未初始化。)
三、類加載器。
類加載器就是負(fù)責(zé)將.class文件加載到內(nèi)存中,并為之生成對應(yīng)的java.lang.Class對象,它負(fù)責(zé)加載所有的類,而一旦一個類被加載入JVM中,就不會被再次載入了。
在Java中,一個類用其全限定類名(即包名+類名)作為標(biāo)識。
而在JVM中,一個類用其全限定類名和其類加載器作為標(biāo)識。
JVM運(yùn)行時會產(chǎn)生3個ClassLoader,分別為:BootstrapClassLoader(根類加載器)、ExtClassLoader(擴(kuò)展類加載器)和AppClassLoader(系統(tǒng)類加載器)。UML結(jié)構(gòu)如下:
其中,BootstrapClassLoader負(fù)責(zé)加載JRE的核心類庫,它不是ClassLoader的子類,使用C++編寫,因此我們在Java中看不到它,通過其子類的getParent()方法獲取時,將返回null。BootstrapClassLoader負(fù)責(zé)裝載JRE目標(biāo)下的rt.jar、charsets.jar等Java核心類庫。
如圖可知,ExtClassLoader和AppClassLoader為ClassLoader的子類。在API中看不到它們,他們位于rt.jar文件中。全限定類名分別為:
sun.misc.Launcher$ExtClassLoader 和 sun.misc.Launcher$AppClassLoader.
其中,ExtClassLoader負(fù)責(zé)裝載JRE擴(kuò)展目錄ext中JAR包,而AppClassLoader負(fù)責(zé)裝載Classpath路徑下的類包。
測試如下:
package com.stopTalking.crazy;
public class TestClassLoader {
public static void main(String[] args) {
//獲取當(dāng)前線程的類裝載器
ClassLoader loader = Thread.currentThread().getContextClassLoader();
//獲取System類的類裝載器
ClassLoader loader1 = System.class.getClassLoader();
//獲取本類TestClassLoader的類裝載器loader2
ClassLoader loader2 = TestClassLoader.class.getClassLoader();
//獲取loader2的父類
ClassLoader loader3 = loader2.getParent();
//獲取loader2的父類的父類
ClassLoader loader4 = loader3.getParent();
System.out.println(loader);
System.out.println(loader1);
System.out.println(loader2);
System.out.println(loader3);
System.out.println(loader4);
}
}
控制臺輸出:
//當(dāng)前線程類獲取的類加載器是AppClassLoader
sun.misc.Launcher$AppClassLoader@6b97fd
//System類為根裝載器加載,java中訪問不到,所以為null
null
//本類的類加載器當(dāng)然也是AppClassLoader
sun.misc.Launcher$AppClassLoader@6b97fd
sun.misc.Launcher$ExtClassLoader@1c78e57
null