從接觸Java以來(lái),寫(xiě)得最多的就是main()函數(shù):
public static void main(String ?arg[]){}
但是以前一直都沒(méi)有問(wèn)自己,為什么要這么寫(xiě),因?yàn)樵赾語(yǔ)言中就沒(méi)有這樣子的要求。其實(shí)這是一個(gè)不需要解釋的問(wèn)題,因?yàn)閖ava標(biāo)準(zhǔn)就是這么規(guī)定的,那么既然是java標(biāo)準(zhǔn)規(guī)定的,我們按照規(guī)定來(lái)執(zhí)行就好了。不過(guò),這并不是一個(gè)很好的學(xué)習(xí)態(tài)度,如果總是知其然而不知其所以然,總會(huì)對(duì)java有種隔膜的感覺(jué)。就是發(fā)現(xiàn)問(wèn)題了,不去解決,不去了解為什么,心里總是會(huì)有牽絆。今天既然又問(wèn)到了這個(gè)問(wèn)題,就搞懂記錄下來(lái)。
在java中,main()方法是java應(yīng)用程序的入口方法。java虛擬機(jī)通過(guò)main方法找到需要啟動(dòng)的運(yùn)行程序,并且檢查main函數(shù)所在類是否被java虛擬機(jī)裝載。如果沒(méi)有裝載,那么就裝載該類,并且裝載所有相關(guān)的其他類。因此程序在運(yùn)行的時(shí)候,第一個(gè)執(zhí)行的方法就是main()方法。通常情況下, 如果要運(yùn)行一個(gè)類的方法,必須首先實(shí)例化出來(lái)這個(gè)類的一個(gè)對(duì)象,然后通過(guò)"對(duì)象名.方法名()"的方式來(lái)運(yùn)行方法,但是因?yàn)閙ain是程序的入口,這時(shí)候還沒(méi)有實(shí)例化對(duì)象,因此將main方法聲明為static的,這樣這個(gè)方法就可以直接通過(guò)“類名.方法名()”的方式來(lái)調(diào)用。
例如:虛擬機(jī)通過(guò)調(diào)用某個(gè)指定類的方法main啟動(dòng),傳遞給main一個(gè)字符串?dāng)?shù)組參數(shù),使指定的類被裝載,同時(shí)鏈接該類所使用的其它的類型,并且初始化它們。例如對(duì)于程序:
public class HelloApp {
public static void main(String[] args) {
System.out.println("Hello World!");
for (int i = 0; i < args.length; i++) {
System.out.println(args);
}
}
}
編譯后在命令行模式下鍵入: java HelloApp run virtual machine
將通過(guò)調(diào)用HelloApp的方法main來(lái)啟動(dòng)java虛擬機(jī),傳遞給main一個(gè)包含三個(gè)字符串"run"、"virtual"、"machine"的數(shù)組。現(xiàn)在我們略述虛擬機(jī)在執(zhí)行HelloApp時(shí)可能采取的步驟。
開(kāi)始試圖執(zhí)行類HelloApp的main方法,發(fā)現(xiàn)該類并沒(méi)有被裝載,也就是說(shuō)虛擬機(jī)當(dāng)前不包含該類的二進(jìn)制代表,于是虛擬機(jī)使用ClassLoader試圖尋找這樣的二進(jìn)制代表。如果這個(gè)進(jìn)程失敗,則拋出一個(gè)異常。類被裝載后同時(shí)在main方法被調(diào)用之前,必須對(duì)類HelloApp與其它類型進(jìn)行鏈接然后初始化。鏈接包含三個(gè)階段:檢驗(yàn),準(zhǔn)備和解析。檢驗(yàn)檢查被裝載的主類的符號(hào)和語(yǔ)義,準(zhǔn)備則創(chuàng)建類或接口的靜態(tài)域以及把這些域初始化為標(biāo)準(zhǔn)的默認(rèn)值,解析負(fù)責(zé)檢查主類對(duì)其它類或接口的符號(hào)引用,在這一步它是可選的。類的初始化是對(duì)類中聲明的靜態(tài)初始化函數(shù)和靜態(tài)域的初始化構(gòu)造方法的執(zhí)行。一個(gè)類在初始化之前它的父類必須被初始化。整個(gè)過(guò)程如下: