Spring原始碼學習之:ClassLoader學習(2)

Love Lenka發表於2016-11-07

轉載:http://longdick.iteye.com/blog/332580

大家都知道一個java應用專案可以打包成一個jar,當然你必須指定一個擁有main函式的main class作為你這個jar包的程式入口。

 

具體的方法是修改jar包內目錄META-INF下的MANIFEST.MF檔案。

 

比如有個叫做test.jar的jar包,裡面有一個擁有main函式的main class:test.someClassName

我們就只要在MANIFEST.MF裡面新增如下一句話:

Main-Class: test.someClassName

 

 然後我們可以在控制檯裡輸入java -jar  test.jar即可以執行這個jar。

 

但是我們這個專案需要引用其他第三方的jar包,在eclipse裡面以專案jar包的形式引用了這個叫做some.jar的包,當時放在專案的 lib子目錄下,最後專案打包時把這個some.jar也打進來了,但是用java -jar執行這個test.jar的時候報找不到Class異常,原因就是jar引用不到放在自己內部的jar包。

 

那怎麼辦?

 

執行時將其加入classpath的方式行不行?就是在執行jar的同時加入classpath引數:

java -classpath some.jar  -jar test.jar

 

這種方式是不行的,因為使用classpath指定的jar是由AppClassloader來載入,java 命令 加了-jar 引數以後,AppClassloader就只關注test.jar範圍內的class了,classpath引數失效。

 

那該怎麼引用其他的jar包呢?

 

方法一、使用Bootstrap Classloader來載入這些類。

 

我們可以在執行時使用如下引數:

 

 

-Xbootclasspath:完全取代系統Java classpath.最好不用。
-Xbootclasspath/a: 在系統class載入後載入。一般用這個。
-Xbootclasspath/p: 在系統class載入前載入,注意使用,和系統類衝突就不好了.

win32     java -Xbootclasspath/a: some.jar;some2.jar;  -jar test.jar
unix          java -Xbootclasspath/a: some.jar:some2.jar:  -jar test.jar
win32系統每個jar用分號隔開,unix系統下用冒號隔開

 

 

 

方法二、使用Extension Classloader來載入

 

你可以把需要載入的jar都扔到%JRE_HOME%/lib/ext下面,這個目錄下的jar包會在Bootstrap Classloader工作完後由Extension Classloader來載入。非常方便,非常省心。:)

 

 

 

方法三、還是用AppClassloader來載入,不過不需要classpath引數了

 

我們在MANIFEST.MF中新增如下程式碼:

Class-Path: lib/some.jar

 

lib是和test.jar同目錄的一個子目錄,test.jar要引用的some.jar包就在這裡面。

然後測試執行,一切正常!

 

如果有多個jar包需要引用的情況:

Class-Path: lib/some.jar lib/some2.jar

每個單獨的jar用空格隔開就可以了。注意使用相對路徑。

 

另:如果META-INF 下包含INDEX.LIST檔案的話,可能會使Class-Path配置失效。INDEX.LIST是Jar打包工具打包時生成的索引檔案,刪除對執行不產生影響。

 

 

方法四、自定義Classloader來載入

這種方法是終極解決方案,基本上那些知名java應用都是那麼幹的,如tomcat、jboss等等。

這種方式有點複雜,需要專門開貼討論。關於ClassLoader的原理和自定義ClassLoader可以參考這篇  http://longdick.iteye.com/blog/442213。

 

 

總結:

以上四種方法都可以用,特別是程式執行在非常單純的環境中時。但是,如果是執行在多工,多應用的環境中時,最好每個應用都能相互獨立,第一種和第二種方案都有可能對其他應用產生影響,因此最好就是選擇第三種和第四種。

相關文章