Java 獲取 jar包以外的資源操作
在使用 jar 執(zhí)行 java 代碼時(shí),有一個(gè)需求是從 jar 包所在目錄的同級(jí)目錄下讀取配置文件的需求,從網(wǎng)上找了很多方法感覺都挺復(fù)雜的,
在這里總結(jié)一下.
以classpath 開頭的 URL 表示該文件為jar包內(nèi)文件的路徑.
如:classpath://config/app.config表示jar包根路徑config文件夾下的app.config文件
以file開頭的URL表示該文件為jar 包外文件的路徑
如:file://./config/app.config表示
摘要
// 當(dāng)前我想從jar包的同級(jí)目錄下讀取一個(gè)名為 'config.txt'的文件的話,我需要指定目錄為.
File file = new File(".","config.txt")
說明
File file = new File("config.txt")
當(dāng)只包含文件名稱時(shí),java程序會(huì)默認(rèn)嘗試從jar包的根路徑去讀取文件,當(dāng)嘗試使用 file.getCanonicalPath() 方法讀取時(shí),便會(huì)得到該文件在jar包內(nèi)的路徑.
示例
我當(dāng)前的工程的路徑為D:\WorkSpace\path_demo01\
在工程執(zhí)行以下java代碼:
當(dāng)指定parent時(shí):
會(huì)從parent下查找path資源.
log(FileUtil.file(".", "/config/app.config").getCanonicalPath()); log(FileUtil.file(".", "config/app.config").getCanonicalPath()); // D:\WorkSpace\path_demo01\config\app.config //加載與當(dāng)前jar包同級(jí)目錄下的文件 log(FileUtil.file(".", "app.config").getCanonicalPath());
當(dāng)沒有指定parent:
如果path為絕對路徑時(shí),會(huì)從絕對路徑下查找
如果path為相對路徑時(shí),會(huì)從classpath的根路徑下開始查找
log(FileUtil.file("/config/app.config").getCanonicalPath()); // D:\config\app.config log(FileUtil.file("config/app.config").getCanonicalPath()); // D:\WorkSpace\path_demo01\target\classes\config\app.config
通過當(dāng)前類加載資源:
如果path為相對路徑會(huì)指定要加載的資源路徑與當(dāng)前類所在包的路徑一致
如果path為絕對路徑,那么就會(huì)從classpath的根路徑下開始查找
log(App.class.getResource("/config/app.config")); // file:/D:/WorkSpace/path_demo01/target/classes/config/app.config log(App.class.getResource("config/app.config")); // file:/D:/WorkSpace/path_demo01/target/classes/top/ghimi/config/app.config
通過類加載器加載資源:
默認(rèn)是從ClassPath根下獲取,path不能以/開頭,最終是由ClassLoader獲取資源.
log(App.class.getClassLoader().getResource("/config/app.config")); // null log(App.class.getClassLoader().getResource("config/app.config")); // file:/D:/WorkSpace/path_demo01/target/classes/config/app.config
加載jar包內(nèi)的資源
當(dāng)代碼打包成jar包的形式后,是無法通過new File()的形式加載jar包內(nèi)的資源的.此時(shí)有可能拋出
FileNotFoundException異常,是由于將path當(dāng)成jar包外的目錄查找不到資源導(dǎo)致的.
URI is not hierarchical異常,是由于無法直接讀取jar包中資源(透明)而拋出的異常.
解決方法:
使用 getResourceAsStream()方法直接獲取資源的流而不是getResource()獲取資源文件對象的方式讀取資源.
//修改前,未打包成jar包時(shí)能夠正常執(zhí)行,打包后會(huì)拋出異常 log(App.class.getClassLoader().getResource("config/app.config")); // 修改后,打成jar包后也可以正常加載資源 log(App.class.getClassLoader().getResourceAsStream("config/app.config"));
加載jar包外的資源
會(huì)從parent目錄下查找path資源.
//加載與當(dāng)前jar包同級(jí)目錄下的文件 log(FileUtil.file(".", "app.config").getCanonicalPath()); // D:\WorkSpace\path_demo01\app.config //加載與當(dāng)前jar包的上一級(jí)目錄下的文件 log(FileUtil.file("..", "app.config").getCanonicalPath()); // D:\WorkSpace\app.config
借用工具h(yuǎn)utool:
<dependency> <groupId>cn.hutool</groupId> <artifactId>hutool-all</artifactId> <version>4.5.6</version> </dependency>
補(bǔ)充知識(shí):java中jar包內(nèi)的類訪問jar包內(nèi)部的資源文件的路徑問題
在本地項(xiàng)目中,若我們要訪問項(xiàng)目中的資源文件,則一般使用相對路徑或者用System.getProperities("user.dir")得到項(xiàng)目根目錄,然后再訪問資源文件,但是在將該工程和資源文件打包為jar包,運(yùn)行該jar文件時(shí),會(huì)顯示找不到資源文件的錯(cuò)誤。
在如下項(xiàng)目結(jié)構(gòu)樹中,項(xiàng)目根目錄為nlpir,如果我們要在src下的某個(gè)package的某個(gè)java文件中訪問blackWhite文件夾中的文件,則相對路徑為"blackWhite/....."即可。但是在打包為jar包時(shí),即使我們把blackWhite文件夾同樣加入到打包的文件行列,在運(yùn)行該jar包時(shí),會(huì)出錯(cuò):找不到blackWhite中某文件的路徑。
解決方法:
使用Class.getResource或者是ClassLoader.getResourceAsStream()將文件內(nèi)容放到InputStream中,具體使用如下:
String s1 = this.getClass().getResource("/library.properties").getPath();
或者為:
String s1 = CodeTest.class.getResource("/library.properties").getPath();
注意,使用class的getRescource時(shí),要注意路徑前要加"/",即根目錄,此處的根目錄是src
若像如下使用:
String class_str = this.getClass().getResource("logback.xml").getPath();
則會(huì)出錯(cuò)如下:
使用ClassLoader時(shí),如下:
this.getClass().getClassLoader().getResource()
在使用ClassLoader時(shí),路徑前面不能加"/",使用相對路徑。
如下示例:
@Test public void test4(){ String class_str = this.getClass().getResource("/logback.xml").getPath(); String class_str2 = TempTest.class.getResource("/logback.xml").getPath(); String classLoader_str = this.getClass().getClassLoader().getResource("logback.xml").getPath(); InputStream is = this.getClass().getClassLoader().getResourceAsStream("logback.xml"); System.out.println(class_str); System.out.println(class_str2); System.out.println(classLoader_str); System.out.println(is == null ); }
結(jié)果如下:
String ss = TempTest.class.getResource("/").getPath();
上述該代碼得到的是項(xiàng)目的根目錄,即nlpir的根目錄,結(jié)果如下:
/C:/eclipse/eclipse/workspace/nlpir/out/production/nlpir/
如下代碼:
@Test public void readProperties(){ String ss = TempTest.class.getResource("/").getPath(); System.out.println(ss); String s = new File(ss).getParentFile().getPath(); System.out.println(s); String system_str = System.getProperty("user.dir"); System.out.println(system_str); }</span>
運(yùn)行結(jié)果如下:
其中,F(xiàn)ile.getParentFile()可用于求父目錄
將上述readProperties函數(shù)打包為jar包在命令行使用java -jar TempTest.jar運(yùn)行時(shí),結(jié)果如下:
由此可見,打包成jar包時(shí)和在ide中直接運(yùn)行的結(jié)果并不一樣,所以在jar包中的class類要訪問自己jar包中的資源文件時(shí),應(yīng)該使用Class.getResource或者是getResourceAsStream放在InputStream中,再進(jìn)行訪問。但是該方法只能訪問到src下的資源文件,因?yàn)槠涓夸泴?yīng)的就是src,無法訪問到項(xiàng)目根目錄下src外的文件,如上述項(xiàng)目結(jié)構(gòu)圖中的blackWhite中的文件無法訪問到,解決方法還木有找到。。。。。。
當(dāng)jar包外部的類需要訪問某個(gè)jar包的資源文件時(shí),使用JarFile類,
具體使用方法如下:
如果你對于常用的ZIP格式比較熟悉的話,JAR文件也就差不多。JAR文件提供一種將多個(gè)文件打包到一個(gè)文件中的方法,其中每一個(gè)文件可能獨(dú)立地被壓縮。JAR文件所增加的內(nèi)容是manifest,它允許開發(fā)者可以提供附加的關(guān)于內(nèi)容的信息。例如,manifest表明JAR文件中的哪個(gè)文件是用來運(yùn)行一個(gè)程序的,或者庫的版本號(hào)等。
J2SEDK提供了一個(gè)JAR工具,你可以用它從控制臺(tái)讀寫JAR文件。然而,如果你需要在程序中代碼讀寫JAR文件,可能需要一點(diǎn)時(shí)間(本文只包含如何在程序中讀寫JAR文件)。好消息是你可以做到這一點(diǎn),而且你不用擔(dān)心解壓的事,因?yàn)轭悗鞂椭阃瓿蛇@些。
首先,通過把將JAR文件位置傳給構(gòu)造函數(shù),創(chuàng)建一個(gè)JarFile的實(shí)例,位置可能是String或File的形式,如下:
JarFile jf = new JarFile("C:/jxl.jar");
或者為:
File file = new File("C:/jxl.jar");
JarFile jarFile = new JarFile(file);
你可能注意到當(dāng)文件不在class path中時(shí),JarFile類對于從JAR中讀取文件文件是很有用的。當(dāng)你想指定目標(biāo)JAR文件時(shí),JarFile類對于從JAR中讀取文件同樣也很有用。
當(dāng)然,如果JAR文件在class path中,從其中讀取文件的方法比較簡單,你可以用下面的方法:
URL url = ClassLoader.getSystemResource(name);
或者為:
InputStream stream =
ClassLoader.getSystemResourceAsStream("javax/servlet/LocalStrings_fr.properties");
當(dāng)你有了該JAR文件的一個(gè)引用之后,你就可以讀取其文件內(nèi)容中的目錄信息了。JarFile的entries方法返回所有entries的枚舉集合 (Enumeration)。通過每一個(gè)entry,你可以從它的manifest文件得到它的屬性,任何認(rèn)證信息,以及其他任何該entry的信息,如它的名字或者大小等。
Enumeration enu = jf.entries(); while (enu.hasMoreElements()) { JarEntry element = (JarEntry) enu.nextElement(); String name = element.getName(); Long size = element.getSize(); Long time = element.getTime(); Long compressedSize = element.getCompressedSize(); System.out.print(name+"/t"); System.out.print(size+"/t"); System.out.print(compressedSize+"/t"); System.out.println(new SimpleDateFormat("yyyy-MM-dd").format(new Date(time))); }
為了從JAR文件中真正讀取一個(gè)指定的文件,你必須到其entry的InputStream。這和JarEntry不一樣。這是因?yàn)镴arEntry只是包含該entry的有關(guān)信息,但是并不實(shí)際包含該entry的內(nèi)容。這和File和FileInputStream的區(qū)別有點(diǎn)兒相似。訪問文件沒有打開文件,它只是從目錄中讀取了該文件的信息。
下面是如何得到entry的InputStream:
InputStream input = jarFile.getInputStream(entry);
當(dāng)你有了輸入流,你就可以像讀取其他流一樣讀取它。在文本流中(text stream),記得使用讀取器(Reader)從流中取得字符。對于面向字節(jié)的流,如圖片文件,直接讀取就行了。
示例:
下面的程序演示如何從JAR文件中讀取文件。指定JAR文件的名稱,要讀取的文件的名稱(打包JAR文件中的某一個(gè)文件)作為參數(shù)來調(diào)用該程序。要讀取的文件應(yīng)該有一個(gè)文本類型的。
import java.io.*; import java.util.jar.*; public class JarRead { public static void main (String args[]) throws IOException { if (args.length != 2) { System.out.println( "Please provide a JAR filename and file to read"); System.exit(-1); } JarFile jarFile = new JarFile(args[0]); JarEntry entry = jarFile.getJarEntry(args[1]); InputStream input = jarFile.getInputStream(entry); process(input); jarFile.close(); } private static void process(InputStream input) throws IOException { InputStreamReader isr = new InputStreamReader(input); BufferedReader reader = new BufferedReader(isr); String line; while ((line = reader.readLine()) != null) { System.out.println(line); } reader.close(); } }
假設(shè)在myfiles.jar文件中有一個(gè)spider.txt文件,spider文件的內(nèi)容如下:
The itsy bitsy spider Ran up the water spout Down came the rain and Washed the spider out
可以通過下面的命令在命令行來顯示該文本文件的內(nèi)容:
java JarRead myfiles.jar spider.txt
以上這篇Java 獲取 jar包以外的資源操作就是小編分享給大家的全部內(nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
java 實(shí)現(xiàn)發(fā)短信功能---騰訊云短信
如今發(fā)短信功能已經(jīng)成為互聯(lián)網(wǎng)公司的標(biāo)配,接下來通過本文給大家介紹java 實(shí)現(xiàn)發(fā)短信功能---騰訊云短信 ,需要的朋友可以參考下2019-08-08Java 添加、修改、讀取、復(fù)制、刪除Excel批注的實(shí)現(xiàn)
這篇文章主要介紹了Java 添加、修改、讀取、復(fù)制、刪除Excel批注的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02Intellij IDEA中如何查看maven項(xiàng)目中所有jar包的依賴關(guān)系圖
這篇文章主要介紹了Intellij IDEA中如何查看maven項(xiàng)目中所有jar包的依賴關(guān)系圖,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05使用maven開發(fā)springboot項(xiàng)目時(shí)pom.xml常用配置(推薦)
這篇文章主要介紹了使用maven開發(fā)springboot項(xiàng)目時(shí)的pom.xml常用配置,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-01-01java必學(xué)必會(huì)之this關(guān)鍵字
java必學(xué)必會(huì)之this關(guān)鍵字,java中this的用法進(jìn)行了詳細(xì)的分析介紹,感興趣的小伙伴們可以參考一下2015-12-12Spring?Boot中使用Swagger3.0.0版本構(gòu)建RESTful?APIs的方法
Swagger?是一個(gè)規(guī)范和完整的框架,用于生成、描述、調(diào)用和可視化?RESTful?風(fēng)格的?Web?服務(wù),這篇文章主要介紹了Spring?Boot中使用Swagger3.0.0版本構(gòu)建RESTful?APIs的方法,需要的朋友可以參考下2022-11-11Java解決xss轉(zhuǎn)義導(dǎo)致轉(zhuǎn)碼的問題
跨站腳本攻擊XSS是最普遍的Web應(yīng)用安全漏洞,本文主要介紹了Java解決xss轉(zhuǎn)義導(dǎo)致轉(zhuǎn)碼的問題,具有一定的參考價(jià)值,感興趣的可以了解一下2023-08-08如何解決getReader() has already been called&
這篇文章主要介紹了如何解決getReader() has already been called for this request問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-05-05Java開發(fā)如何把數(shù)據(jù)庫里的未付款訂單改成已付款
這篇文章主要介紹了Java開發(fā)如何把數(shù)據(jù)庫里的未付款訂單改成已付款,先介紹MD5算法,簡單的來說,MD5能把任意大小、長度的數(shù)據(jù)轉(zhuǎn)換成固定長度的一串字符,實(shí)現(xiàn)思路非常簡單需要的朋友可以參考下2022-11-11