天天看點

Spring源碼學習之:ClassLoader學習(2)

轉載: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等等。

<a href="http://longdick.iteye.com/blog/442213">http://longdick.iteye.com/blog/442213。</a>

總結:

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

繼續閱讀