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:WorkSpacepath_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:WorkSpacepath_demo01configapp.config//加載與當(dāng)前jar包同級(jí)目錄下的文件log(FileUtil.file('.', 'app.config').getCanonicalPath());
當(dāng)沒有指定parent:
如果path為絕對(duì)路徑時(shí),會(huì)從絕對(duì)路徑下查找
如果path為相對(duì)路徑時(shí),會(huì)從classpath的根路徑下開始查找
log(FileUtil.file('/config/app.config').getCanonicalPath());// D:configapp.configlog(FileUtil.file('config/app.config').getCanonicalPath());// D:WorkSpacepath_demo01targetclassesconfigapp.config
通過當(dāng)前類加載資源:
如果path為相對(duì)路徑會(huì)指定要加載的資源路徑與當(dāng)前類所在包的路徑一致
如果path為絕對(duì)路徑,那么就會(huì)從classpath的根路徑下開始查找
log(App.class.getResource('/config/app.config'));// file:/D:/WorkSpace/path_demo01/target/classes/config/app.configlog(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'));// nulllog(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()獲取資源文件對(duì)象的方式讀取資源.
//修改前,未打包成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:WorkSpacepath_demo01app.config//加載與當(dāng)前jar包的上一級(jí)目錄下的文件log(FileUtil.file('..', 'app.config').getCanonicalPath());// D:WorkSpaceapp.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)目中的資源文件,則一般使用相對(duì)路徑或者用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文件夾中的文件,則相對(duì)路徑為'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í),路徑前面不能加'/',使用相對(duì)路徑。
如下示例:
@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)槠涓夸泴?duì)應(yīng)的就是src,無法訪問到項(xiàng)目根目錄下src外的文件,如上述項(xiàng)目結(jié)構(gòu)圖中的blackWhite中的文件無法訪問到,解決方法還木有找到。。。。。。
當(dāng)jar包外部的類需要訪問某個(gè)jar包的資源文件時(shí),使用JarFile類,
具體使用方法如下:
如果你對(duì)于常用的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類對(duì)于從JAR中讀取文件文件是很有用的。當(dāng)你想指定目標(biāo)JAR文件時(shí),JarFile類對(duì)于從JAR中讀取文件同樣也很有用。
當(dāng)然,如果JAR文件在class path中,從其中讀取文件的方法比較簡(jiǎn)單,你可以用下面的方法:
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)從流中取得字符。對(duì)于面向字節(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包以外的資源操作就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. python如何寫個(gè)俄羅斯方塊2. JAVA抽象類及接口使用方法解析3. 《CSS3實(shí)戰(zhàn)》筆記--漸變?cè)O(shè)計(jì)(一)4. 如何通過vscode運(yùn)行調(diào)試javascript代碼5. Python結(jié)合百度語音識(shí)別實(shí)現(xiàn)實(shí)時(shí)翻譯軟件的實(shí)現(xiàn)6. python b站視頻下載的五種版本7. Java構(gòu)建JDBC應(yīng)用程序的實(shí)例操作8. JavaScript設(shè)計(jì)模式之策略模式實(shí)現(xiàn)原理詳解9. XML基本概念XPath、XSLT與XQuery函數(shù)介紹10. vue之elementUi的el-select同時(shí)獲取value和label的三種方式
