java打jar包的幾種方式詳解
一、制作只含有字節(jié)碼文件的jar包
我們先來(lái)看只含有字節(jié)碼文件,即只含有class文件的jar包怎么制作,這是最簡(jiǎn)單的形式
1、最簡(jiǎn)單的jar包——直接輸出hello
最終生成的jar包結(jié)構(gòu)
META-INF
Hello.class
方法步驟
(1)用記事本寫一個(gè)Hello.java的文件
class Hello{
public static void main(String[] agrs){
System.out.println("hello");
}
}
(2)用命令行進(jìn)入到該目錄下,編譯這個(gè)文件
javac Hello.java
(3)將編譯后的Hello.class文件打成jar包
jar -cvf hello.jar Hello.class
c表示要?jiǎng)?chuàng)建一個(gè)新的jar包,v表示創(chuàng)建的過(guò)程中在控制臺(tái)輸出創(chuàng)建過(guò)程的一些信息,f表示給生成的jar包命名
(4)運(yùn)行jar包
java -jar hello.jar 這時(shí)會(huì)報(bào)如下錯(cuò)誤 hello.jar中沒(méi)有主清單屬性
添加Main-Class屬性
用壓縮軟件打開(kāi)hello.jar,會(huì)發(fā)現(xiàn)里面多了一個(gè)META-INF文件夾,里面有一個(gè)MENIFEST.MF的文件,用記事本打開(kāi)
Manifest-Version: 1.0
Created-By: 1.8.0_121 (Oracle Corporation)
在第三行的位置寫入 Main-Class: Hello (注意冒號(hào)后面有一個(gè)空格,整個(gè)文件最后有一行空行),保存
再次運(yùn)行 java -jar hello.jar ,此時(shí)成功在控制臺(tái)看到 hello ,成功
2、含有兩個(gè)類的jar包——通過(guò)調(diào)用輸出hello
最終生成的jar包結(jié)構(gòu)
META-INF
Tom.class
Hello.class
方法步驟
(1)用記事本寫一個(gè)Hello.java和一個(gè)Tom.java的文件
目的是讓Hello調(diào)用Tom的speak方法
class Hello{ public static void main(String[] agrs){ Tom.speak(); } } class Tom{ public static void speak(){ System.out.println("hello"); } }
(2)編譯: javac Hello.java
此時(shí)Hello.java和Tom.java同時(shí)被編譯,因?yàn)镠ello中調(diào)用了Tom,在編譯Hello的過(guò)程中發(fā)現(xiàn)還需要編譯Tom
(3)打jar包,這次我們換一種方式直接定義Main-Class。
Manifest-Version: 1.0
Created-By: 1.8.0_121 (Oracle Corporation)
Main-Class: Hello
事先準(zhǔn)備好上述的MENIFEST.MF文件,并存放在META-INF文件夾下,此時(shí)打jar包的命令如下
jar -cvfm hello.jar META-INF\MENIFEST.MF Hello.class Tom.class
該命令表示用第一個(gè)文件當(dāng)做MENIFEST.MF文件,hello.jar作為名稱,將Hello.class和Tom.class打成jar包。其中多了一個(gè)參數(shù)m,表示要定義MENIFEST文件
(4)運(yùn)行 java -jar hello.jar ,此時(shí)成功在控制臺(tái)看到 hello ,成功
3、有目錄結(jié)構(gòu)的jar包——通過(guò)引包并調(diào)用輸出hello
最終生成的jar包結(jié)構(gòu)
META-INF
com
Tom.class
Hello.class
我們將上一個(gè)稍稍變化一下,將Tom這個(gè)類放在com包下,源文件目錄結(jié)構(gòu)變成
com
Tom.java
Hello.java
同時(shí)Tom.java需要在第一行聲明自己的包名
package com;
Hello.java需要引入Tom這個(gè)類,同樣要在第一行進(jìn)行import
import com.Tom;
方法步驟
(1)編譯Hello.java
(2)打jar包,同樣準(zhǔn)備好MENIFEST文件
jar -cvfm hello.jar META-INF\MENIFEST.MF Hello.class com
注意,最后一個(gè)com表示把com這個(gè)文件夾下的所有文件都打進(jìn)jar包
(3)運(yùn)行 java -jar hello.jar ,此時(shí)成功在控制臺(tái)看到 hello ,成功
(4)優(yōu)化過(guò)程
我們注意到,com包下是有Tom.java源文件的,也被打進(jìn)了jar包里,這樣不太好,能不能優(yōu)化一下javac命令,使所有的編譯后文件編譯到另一個(gè)隔離的地方呢,答案是可以的。
在編譯Hello.java時(shí),先新建一個(gè)target文件夾。然后我們用如下命令
javac Hello.java -d target
該命令表示,將所有編譯后的文件,都放到target文件夾下。
將META-INF文件夾也復(fù)制到target目錄下,進(jìn)入這個(gè)目錄,輸入如下命令
jar -cvfm hello.jar META-INF\MENIFEST.MF *
注意最后一個(gè)位置變成了*,表示把當(dāng)前目錄下所有文件都打在jar包里
優(yōu)化完畢
至此,我們可以總結(jié)出,制作一個(gè)只含有class字節(jié)碼文件的jar包,以下命令足以
javac 要編譯的文件 -d 目標(biāo)位置
jar -cvfm 命名 MENIFEST文件 要打包的文件1 要打包的文件2
二、制作含有jar文件的jar包
我們將場(chǎng)景稍稍變得復(fù)雜一點(diǎn),看看jar包中需要引入其他jar包的場(chǎng)景
1、兩個(gè)jar包間相互調(diào)用——調(diào)用jar外的jar輸出hello
最終生成的jar包結(jié)構(gòu)
hello.jar
tom.jar
方法步驟
準(zhǔn)備:將上述一中寫好的那個(gè)不帶包的tom.jar復(fù)制過(guò)來(lái)(目的是調(diào)用里面的speak方法)
(1)編寫一個(gè)Hello.java并將其編譯成Hello.class,注意,由于Hello里面引用了Tom類的speak方法,因此在打jar包時(shí)應(yīng)使用-cp參數(shù),將tom.jar包引入
javac -cp tom.jar Hello.class
這里的 -cp 表示 -classpath,指的是把tom.jar加入classpath路徑下
(2)將hello.class達(dá)成jar包,步驟略
(3)此時(shí)運(yùn)行 java -jar 發(fā)現(xiàn)報(bào)錯(cuò) ClassNotFoundException:Tom
原因很簡(jiǎn)單,引入jar包需要在MENIFEST.MF文件中配置一個(gè)新屬性:Class-Path,路徑指向你需要的所有jar包
現(xiàn)在MENIFEST.MF這個(gè)文件應(yīng)該變成
Manifest-Version: 1.0
Created-By: 1.8.0_121 (Oracle Corporation)
Main-Class: Hello
Class-Path: Tom.jar
(4)好了,修改這個(gè)文件,再次運(yùn)行,發(fā)現(xiàn)成功在控制臺(tái)輸出 hello
tips:引入多個(gè)jar包,中間用空格隔開(kāi)
至此,我們可以總結(jié)出,命令變化如下
javac -cp xxx.jar 要編譯的文件 -d 目標(biāo)位置
jar -cvfm 命名 MENIFEST文件 要打包的文件1 要打包的文件2
2、jar包中含有jar包——調(diào)用jar內(nèi)的jar輸出hello
最終生成的jar包結(jié)構(gòu)
META-INF
Hello.class
tom.jar
當(dāng)項(xiàng)目中我們把所需要的第三方j(luò)ar包也打進(jìn)了我們自己的jar包中時(shí),如果仍然按照上述操作方式,會(huì)報(bào)找不到Class異常。原因就是jar引用不到放在自己內(nèi)部的jar包。
這種情況的具體實(shí)現(xiàn)細(xì)節(jié)比較復(fù)雜,我會(huì)在后一篇介紹一些知名的java應(yīng)用是如何加載jar包的,來(lái)說(shuō)明這種情況。實(shí)現(xiàn)方式的簡(jiǎn)單說(shuō)明,可以先參考這篇文章:
http://www.dbjr.com.cn/article/174315.htm
三、制作含有資源文件的jar包
1、資源文件在jar包內(nèi)部——讀取jar內(nèi)的文件
最終生成的jar包結(jié)構(gòu)
META-INF
Hello.class
text.txt
方法步驟
import java.io.InputStream; import java.io.BufferedReader; import java.io.InputStreamReader; class Hello{ public static void main(String[] args) throws Exception{ Hello hello = new Hello(); InputStream is = hello.getClass().getResourceAsStream("text.txt"); print(is); } /** * 讀取文件,輸出里面的內(nèi)容,通用方法 */ public static void print(InputStream inputStream) throws Exception { InputStreamReader reader = new InputStreamReader(inputStream, "utf-8"); BufferedReader br = new BufferedReader(reader); String s = ""; while ((s = br.readLine()) != null) System.out.println(s); inputStream.close(); } }
2、資源文件在另一個(gè)jar包內(nèi)部——讀取另一個(gè)jar內(nèi)的文件
最終生成的jar包結(jié)構(gòu)
hello.jar
resource.jar
text.txt
方法步驟
同1一樣,只不過(guò)需要在MENIFEST文件中將resource.jar加入classpath
import java.io.InputStream; import java.io.BufferedReader; import java.io.InputStreamReader; class Hello{ public static void main(String[] args) throws Exception{ Hello hello = new Hello(); InputStream is = hello.getClass().getResourceAsStream("text.txt"); print(is); } /** * 讀取文件,輸出里面的內(nèi)容,通用方法 */ public static void print(InputStream inputStream) throws Exception { InputStreamReader reader = new InputStreamReader(inputStream, "utf-8"); BufferedReader br = new BufferedReader(reader); String s = ""; while ((s = br.readLine()) != null) System.out.println(s); inputStream.close(); } }
3、資源文件在jar包外部——讀取jar外的文件
最終生成的jar包結(jié)構(gòu)
hello.jar
text.txt
方法步驟
import java.io.InputStream; import java.io.BufferedReader; import java.io.InputStreamReader; import java.io.FileInputStream; class Hello{ public static void main(String[] args) throws Exception{ Hello hello = new Hello(); InputStream is = new FileInputStream("text.txt"); print(is); } /** * 讀取文件,輸出里面的內(nèi)容,通用方法 */ public static void print(InputStream inputStream) throws Exception { InputStreamReader reader = new InputStreamReader(inputStream, "utf-8"); BufferedReader br = new BufferedReader(reader); String s = ""; while ((s = br.readLine()) != null) System.out.println(s); inputStream.close(); } }
總結(jié)
以上所述是小編給大家介紹的java打jar包的幾種方式詳解,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
如果你覺(jué)得本文對(duì)你有幫助,歡迎轉(zhuǎn)載,煩請(qǐng)注明出處,謝謝!
相關(guān)文章
使用Java對(duì)Hbase操作總結(jié)及示例代碼
這篇文章主要介紹了使用Java對(duì)Hbase進(jìn)行操作總結(jié),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07Java高并發(fā)編程之CAS實(shí)現(xiàn)無(wú)鎖隊(duì)列代碼實(shí)例
這篇文章主要介紹了Java高并發(fā)編程之CAS實(shí)現(xiàn)無(wú)鎖隊(duì)列代碼實(shí)例,在多線程操作中,我們通常會(huì)添加鎖來(lái)保證線程的安全,那么這樣勢(shì)必會(huì)影響程序的性能,那么為了解決這一問(wèn)題,于是就有了在無(wú)鎖操作的情況下依然能夠保證線程的安全,需要的朋友可以參考下2023-12-12Spring循環(huán)依賴實(shí)現(xiàn)過(guò)程揭秘
這篇文章主要介紹了Spring循環(huán)依賴實(shí)現(xiàn)過(guò)程,Spring的解決循環(huán)依賴是有前置條件的,要解決循環(huán)依賴我們首先要了解Spring Bean對(duì)象的創(chuàng)建過(guò)程和依賴注入的方式2023-01-01idea指定maven的settings文件不生效的問(wèn)題解決
本文主要介紹了idea指定maven的settings文件不生效的問(wèn)題解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-06-06SpringBoot?HikariCP配置項(xiàng)及源碼解析
這篇文章主要為大家介紹了SpringBoot?HikariCP配置項(xiàng)及源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02mybatis-flex與springBoot整合的實(shí)現(xiàn)示例
Mybatis-flex提供了簡(jiǎn)單易用的API,開(kāi)發(fā)者只需要簡(jiǎn)單的配置即可使用,本文主要介紹了mybatis-flex與springBoot整合,具有一定的參考價(jià)值,感興趣的可以了解一下2024-01-01Java多線程之Callable接口的實(shí)現(xiàn)
這篇文章主要介紹了Java多線程之Callable接口的實(shí)現(xiàn),Callable和Runnbale一樣代表著任務(wù),區(qū)別在于Callable有返回值并且可以拋出異常。感興趣的小伙伴們可以參考一下2018-08-08