??本文聊一聊類加載器的應(yīng)用。我們知道,java文件經(jīng)過(guò)javac指令的編譯后會(huì)生成class文件,class文件經(jīng)過(guò)java指令被JVM加載之后會(huì)在其內(nèi)部生成描述該class文件的的Class對(duì)象。在這個(gè)過(guò)程中,java指令起的就是類加載的作用。
??Java中的類加載有三種,按加載順序(指的是某class文件需要被加載時(shí)按以下順序?qū)ふ覒?yīng)該加載它的加載器)如下:Bootstrap類加載器、Extension類加載器、Application類加載器。其中Bootstrap負(fù)責(zé)加載系統(tǒng)類,通常是從rt.jar文件中加載。Extension類加載器是從jre/lib/ext目錄下加載“標(biāo)準(zhǔn)的拓展”。也就是說(shuō)這倆類加載器是加載一些Java設(shè)計(jì)者為我們提供的類。而第三個(gè)Application類加載器是加載第三方的jar包和src下我們自己編寫的類。
??除了這些理論性的東西之外,對(duì)于Java EE開發(fā)者來(lái)說(shuō)類加載器最大的作用就是獲得資源文件的類路徑(class文件所在的路徑)。獲得類路徑有三步,獲得字節(jié)碼對(duì)象、獲得字節(jié)碼對(duì)象的類加載器、獲得資源的URL、獲得URL中的路徑部分。
??看完這里例子后其實(shí)我們還有一個(gè)疑點(diǎn):字節(jié)碼對(duì)象應(yīng)該獲取誰(shuí)的?答案是只要是通過(guò)Application類加載器加載的類就可以。比如小碼農(nóng)還有一個(gè)Second類,打印出來(lái)的是相同的效果。當(dāng)然使用的時(shí)候一般使用其所在的類就可以了。
??說(shuō)到這里我們就必須得說(shuō)一個(gè)問(wèn)題,不是所有類都能通過(guò)getClassLoader()方法獲得他的類加載器(獲取不到返回null),只有Extension類加載器和Application類加載器能通過(guò)getClassLoader()方法獲得它的類加載器。為什么呢?因?yàn)锽ootstrap類加載器通常是使用C編寫的,它沒有對(duì)應(yīng)任何的JClassLoader對(duì)象,而ExtClassLoader和AppClassLoader是使用Java寫的Java類。
??到這里,還剩下一個(gè)Extension類加載器沒有演示過(guò),下圖演示一下。不過(guò)Extension類加載器的演示比較困難,小碼農(nóng)在jre/lib/ext下找了好幾個(gè)類都不能使用,所以采取了一種投機(jī)取巧的方法來(lái)演示。首先把我們的測(cè)試程序打包成jar包,然后拷貝到j(luò)re/lib/ext下,這樣再運(yùn)行我們的類時(shí)由于先使用Extension類加載器加載,所以會(huì)運(yùn)行jre/lib/ext的文件。如下圖所示。
??我們獲得資源的相對(duì)路徑一般是為了通過(guò)InputStream讀取,所以還需要距離實(shí)用還有一步讀取。完整的操作如下:
??然而Java的設(shè)計(jì)者就特別貼心了,不用我們這么麻煩的操作,它提供了一個(gè)getResourceAsStream()方法。
??而更貼心的是我們連獲取類加載器這一步都可以省略,通過(guò)Class類對(duì)象就可以直接使用getResourceAsStream(),當(dāng)然它在底層最終使用的還是類加載器的getResourceAsStream()。
??最常用的功能的解析已經(jīng)結(jié)束了,但在Java核心技術(shù)卷中把類記載這部分放在了“安全”這一章,這是由于類加載器在加載類到虛擬機(jī)中時(shí)會(huì)檢查類的完整性。而且所有的類加載器都會(huì)負(fù)責(zé)控制代碼運(yùn)行安全的安全管理器類進(jìn)行合作來(lái)保證Java的安全性。