大家都知道一個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
二、maven項目可直接在pom.xml 中加入如下
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-jar-plugin</artifactId>
<version>2.4</version>
<configuration>
<archive>
<manifest>
<addClasspath>true</addClasspath>
<classpathPrefix>lib/</classpathPrefix>
<mainClass>com.*.SomeClassName</mainClass>
</manifest>
</archive>
</configuration>
</plugin>
然后我們可以在控制台里輸入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參數失效。
方法一、使用Bootstrap Classloader來加載這些類。
-Xbootclasspath:完全取代系統Java classpath.最好不用。
-Xbootclasspath/a: 在系統class加載后加載。一般用這個。
-Xbootclasspath/p: 在系統class加載前加載,注意使用,和系統類沖突就不好了
方法二、使用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。
總結:
以上四種方法都可以用,特別是程序運行在非常單純的環境中時。但是,如果是運行在多任務,多應用的環境中時,最好每個應用都能相互獨立,第一種和第二種方案都有可能對其他應用產生影響,因此最好就是選擇第三種和第四種。