本篇文章為大家展示了如何使用Java擴(kuò)展機(jī)制加載所有JAR包,內(nèi)容簡(jiǎn)明扼要并且容易理解,絕對(duì)能使你眼前一亮,通過(guò)這篇文章的詳細(xì)介紹希望你能有所收獲。
成都創(chuàng)新互聯(lián)公司-成都網(wǎng)站建設(shè)公司,專(zhuān)注成都網(wǎng)站制作、成都做網(wǎng)站、外貿(mào)營(yíng)銷(xiāo)網(wǎng)站建設(shè)、網(wǎng)站營(yíng)銷(xiāo)推廣,域名申請(qǐng),虛擬主機(jī),成都網(wǎng)站托管有關(guān)企業(yè)網(wǎng)站制作方案、改版、費(fèi)用等問(wèn)題,請(qǐng)聯(lián)系成都創(chuàng)新互聯(lián)公司。
Java 擴(kuò)展機(jī)制在Java教程中被描述為一種“通過(guò)標(biāo)準(zhǔn)可擴(kuò)展的方式來(lái)讓Java平臺(tái)上所有應(yīng)用使用自定義API”。正如在理解擴(kuò)展機(jī)制進(jìn)行類(lèi)加載中描述的,“擴(kuò)展框架充分使用了類(lèi)加載代理機(jī)制”。這種機(jī)制會(huì)在rt.jar引導(dǎo)(boot)類(lèi)加載之后,標(biāo)準(zhǔn)classpath中的類(lèi)加載之前,加載擴(kuò)展類(lèi)。
擴(kuò)展目錄的工作機(jī)制在類(lèi)的加載上與classpath有點(diǎn)類(lèi)似。對(duì)Java應(yīng)用程序來(lái)說(shuō),所有擴(kuò)展目錄下JAR文件包含的類(lèi)都可以訪(fǎng)問(wèn)。然而,會(huì)有一些關(guān)鍵的不同點(diǎn)。這些區(qū)別會(huì)在下面的文字中高亮顯示。
特征 | Classpath | 擴(kuò)展機(jī)制(可選包) |
---|---|---|
作用域 | 典型的應(yīng)用相關(guān)
主機(jī)上所有可能的JRE
| 所有運(yùn)行在特定JRE上的JVM
各種主機(jī)上的JRE
|
如何指定 | .jar文件
.class Files
| 所有在指定目錄下的JAR文件都會(huì)被加載(即使擴(kuò)展名不是.jar或者沒(méi)有擴(kuò)展名) |
類(lèi)加載順序 | 引導(dǎo)和擴(kuò)展類(lèi)加載之后 | 引導(dǎo)類(lèi)加載之后,classpath上的類(lèi)加載之前 |
一個(gè)最重要且值得重視的問(wèn)題是,擴(kuò)展機(jī)制會(huì)找出所有jar格式的文件,即使文件后綴名不是.jar。這意味著,改變classpath中的jar文件后綴名以此逃過(guò)通配符的篩選,這種方法在擴(kuò)展目錄中行不通。
我會(huì)用一些簡(jiǎn)單的例子來(lái)展示一些上面提到的區(qū)別。接下來(lái)的兩段代碼是一個(gè)簡(jiǎn)單的HelloWorld類(lèi)和一個(gè)main應(yīng)用程序中的Main類(lèi)。Main通過(guò)調(diào)用main方法來(lái)使用HelloWorld類(lèi)。
HelloWorld.java
public class HelloWorld { @Override public String toString() { return "Hello, World!"; } }
Main.java
import static java.lang.System.out; public class Main { public static void main(final String[] arguments) { out.println(new HelloWorld()); } }
為了展示classpath和擴(kuò)展機(jī)制的主要區(qū)別,我將會(huì)把編譯過(guò)的HelloWorld.class文件歸檔到一個(gè)jar包里,命名為HelloWorld.jar。并把它放在一個(gè)跟編譯過(guò)的Main.class不同的目錄下。
為了展示傳統(tǒng)的classpath的使用,我把HelloWorld.jar放在一個(gè)叫做C:\hello的目錄下并且會(huì)用通配符訪(fǎng)問(wèn)JAR來(lái)給Main使用。下面的兩個(gè)截圖對(duì)此進(jìn)行了展示。
上面的截圖說(shuō)明,當(dāng)某個(gè)類(lèi)是在擴(kuò)展目錄下的某個(gè)JAR里,Java launcher甚至不需要把HelloWorld.class放到同一個(gè)目錄下或者在classpath中指定。這常常被用來(lái)說(shuō)明使用擴(kuò)展機(jī)制的優(yōu)點(diǎn)。因?yàn)樗性谶@個(gè)JRE(或者可能是主機(jī)上的所有應(yīng)用)上運(yùn)行的程序都可以不用在classpath上指定就能看到擴(kuò)展目錄下的類(lèi)。
使用傳統(tǒng)classpath方式——指導(dǎo)應(yīng)用去加載JAR中的類(lèi),包含.class文件的JAR文件必須以.jar結(jié)尾。接下來(lái)的截圖展示了當(dāng)把在 classpath引用的目錄下的HelloWorld.jar重命名為HelloWorld.backup之后所發(fā)生的事情。
***一張截圖展示了,擴(kuò)展目錄下過(guò)時(shí)的HelloWorld類(lèi)優(yōu)先于同一目錄下的新定義的HelloWorld類(lèi)加載。甚至當(dāng)我把當(dāng)前目錄寫(xiě)進(jìn) classpath中,擴(kuò)展目錄下的舊版本的類(lèi)仍然優(yōu)先。接下來(lái)的圖也同樣展示了擴(kuò)展目錄下的JAR文件“隱藏”了更新的JAR以及其中類(lèi)的新方法。這些擴(kuò)展目錄下的JAR文件甚至都不是以.jar結(jié)尾的。
剛剛展示的這個(gè)例子,在擴(kuò)展目錄下JAR導(dǎo)致的眾多問(wèn)題來(lái)說(shuō)不算很復(fù)雜。例子中,至少有一個(gè)NoSuchMethodError來(lái)提醒這個(gè)問(wèn) 題。一個(gè)潛在的更加復(fù)雜的情況是,舊的類(lèi)有和新類(lèi)一樣的方法簽名但實(shí)現(xiàn)的方式已經(jīng)過(guò)時(shí)。在這種情況下,可能沒(méi)有錯(cuò)誤、異?;蛘遲hrowable中任何一種,但是應(yīng)用的邏輯不會(huì)像預(yù)期那樣工作。舊的方法可能會(huì)一直存在代碼的底層直到被發(fā)現(xiàn)。當(dāng)缺乏單元測(cè)試或其他測(cè)試時(shí)尤其如此。
使用擴(kuò)展目錄會(huì)讓開(kāi)發(fā)人員變得輕松。因?yàn)閿U(kuò)展目錄下JAR文件中的類(lèi),可以被所有運(yùn)行在與此擴(kuò)展目錄(如果在操作系統(tǒng)上在主機(jī)范圍內(nèi)啟用擴(kuò)展目錄,那么所有主機(jī)上的JRE都可以訪(fǎng)問(wèn))關(guān)聯(lián)JRE上的應(yīng)用訪(fǎng)問(wèn)。然而,隨意使用擴(kuò)展目錄會(huì)有一定的風(fēng)險(xiǎn)。你會(huì)非常容易忘記擴(kuò)展目錄下過(guò)時(shí)的類(lèi)。這會(huì)妨礙類(lèi)加載器選擇明顯應(yīng)當(dāng)被加載的版本。這種情況下,本來(lái)應(yīng)該讓開(kāi)發(fā)者感覺(jué)輕松的擴(kuò)展機(jī)制會(huì)讓他們非常痛苦。
Elliotte Rusty Harold提對(duì)擴(kuò)展機(jī)制有一個(gè)警告:“盡管這些看上去很方便,從長(zhǎng)遠(yuǎn)來(lái)看也是引入了一個(gè)隱患,遲早你會(huì)從一個(gè)你根本沒(méi)想過(guò)的地方載入一個(gè)錯(cuò)誤的類(lèi)版本。這會(huì)浪費(fèi)你不少時(shí)間調(diào)試”。Java教程同樣提出警告(我在這里也著重強(qiáng)調(diào)):“盡管這個(gè)機(jī)制擴(kuò)展了平臺(tái)的核心API,但是應(yīng)該審慎使用。大部分情況,它是用于像JCP這樣標(biāo)準(zhǔn)化比較好的接口,同時(shí)也適用于整個(gè)站點(diǎn)的接口”。
盡管擴(kuò)展(可選包)機(jī)制與classpath機(jī)制很像,并且它們都用于部分的類(lèi)加載,兩者之間的區(qū)別也是非常值得注意的。特別的,記住所有的在擴(kuò)展目錄下的JAR文件(即使它們沒(méi)有以.jar結(jié)尾)都會(huì)被加載是很重要的。給那些JARs重命名甚至改變他們的文件后綴名都不足以讓類(lèi)加載器忽略它們。另一方面,使用classpath的時(shí)候,重命名classpath中指定的JAR文件會(huì)使該JAR無(wú)法加載,改變后綴名后,即使在classpath中使用通配符也無(wú)法加載所有目錄中的JAR。
一些情況下,擴(kuò)展機(jī)制是比較好的選擇,但是這種情況相當(dāng)少。當(dāng)處理預(yù)期以外的NoSuchMethodErrors問(wèn)題時(shí),記住擴(kuò)展機(jī)制使很重要的。這樣就會(huì)去檢查看看是否問(wèn)題就出在擴(kuò)展的目錄中。
上述內(nèi)容就是如何使用Java擴(kuò)展機(jī)制加載所有JAR包,你們學(xué)到知識(shí)或技能了嗎?如果還想學(xué)到更多技能或者豐富自己的知識(shí)儲(chǔ)備,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。