關(guān)于jar讀取文件路徑的這個(gè)問(wèn)題可愁死我了!
(以下內(nèi)容如有不準(zhǔn)確,或錯(cuò)誤的地方請(qǐng)留言.我會(huì)積極配合領(lǐng)導(dǎo)去查證并更正我的錯(cuò)誤.)
先看項(xiàng)目樹:(對(duì)應(yīng)項(xiàng)目Open in Terminal? ? ---? ?tree >>E:\tree.txt)
當(dāng)我想把項(xiàng)目打成jar包并發(fā)布到服務(wù)器上的時(shí)候,它總是這樣親切的對(duì)我說(shuō):
我苦思冥想,然后.......我就去google了.......
最終以這種方式解決了問(wèn)題:
究其原因,其實(shí)是這個(gè)樣子(...)
打包之后捌臊,jar包中的所加載的文件路徑發(fā)生了變化,我們?cè)诎?~.*(例如:test.json)打包到C盤之后兜材,其路徑變?yōu)閒ile:/C:/*.jar!/~.*理澎,如果你在原項(xiàng)目中使用new File(filePath)之類的方法來(lái)加載的話,肯定會(huì)找不到資源文件曙寡。主要是因?yàn)镴ar包是一個(gè)單獨(dú)的文件而非文件夾糠爬,絕對(duì)不可能通過(guò)file:/C:/.../*.jar!/~.*這種形式的文件URL來(lái)定位~.*。所以即使是相對(duì)路徑举庶,也無(wú)法定位到Jar包內(nèi)的資源文件执隧。
這個(gè)時(shí)候我們需要使用到getResource()和getResourceAsStream();
這兩個(gè)方法沒(méi)有區(qū)別,只是返回的值不一樣.
首先上一段測(cè)試代碼:
它的結(jié)果是這樣的(surname.json是放在resources下面的,打包會(huì)直接打到classes的根目錄下);
我們發(fā)現(xiàn)這樣一個(gè)現(xiàn)象:
NameUtil.class.getResource("surname.json");
和
NameUtil.class.getClassLoader().getResource("surname.json");
結(jié)果是不一樣的....一臉懵逼...
先試著腦補(bǔ)2分鐘.
當(dāng)我們使用NameUtil.class.getResource("surname.json")的時(shí)候Class.java解析出來(lái)的路徑是:
它的解析代碼如下:
其實(shí)就是如果路徑是以"/"開(kāi)頭的,則指絕對(duì)路徑(相對(duì)于項(xiàng)目classes的絕對(duì)路徑!),如果不是........則
是獲取了當(dāng)前類NameUtil(因?yàn)槭荖ameUtil.class.getResource(),所有這里獲取的是NameUtil的命名空間)的類路徑.然后根據(jù)這個(gè)路徑來(lái)作為參照物.對(duì)傳入的path進(jìn)行參照解析.如果你是絕對(duì)則沒(méi)有參照價(jià)值,如果是相對(duì)路徑,則傳入的path就必須與NameUtil的類在同一個(gè)包下才可以正確解析.
我們?cè)賮?lái)看? ?NameUtil.class.getClassLoader().getResource("surname.json");
System.out.println(NameUtil.class.getClassLoader());
它的值是:sun.misc.Launcher$AppClassLoader@18b4aac2
也就是說(shuō)NameUtil的類加載器是AppClassLoader,
(關(guān)于類加載器這里有一篇文章,將的很詳細(xì)!??java中的ClassLoader詳解)
AppClassLoader是classes的默認(rèn)加載器,即咱們的項(xiàng)目的class都是由這家伙加載的.所以它的路徑就是classes的根路徑,但是它沒(méi)有對(duì)"/"做處理...它的絕對(duì)路徑是模糊的,這里對(duì)于"/"只能返回空值.針對(duì)于這方面的問(wèn)題,以后再研究.這里默認(rèn)得到NameUtil.class.getClassLoader().getResource的相對(duì)路徑是根路徑.為了驗(yàn)證這點(diǎn)
System.out.println(NameUtil.class.getClassLoader().getResource("surname.json"));
得到的是這樣的路徑
這里說(shuō)一下? "./"和"/"不一樣,它代表當(dāng)前路徑.
NameUtil.class.getClassLoader().getResource("surname.json")得到的路徑和
NameUtil.class.getResource("/surname.json")是一樣的,所以這里得到一個(gè)等式:
NameUtil.class.getClassLoader().getResource("surname.json")==NameUtil.class.getResource("/surname.json");
到這里,我們讀取文件的路徑問(wèn)題基本算是解決了.
總結(jié):
在我們的IDE中,編寫好的java代碼會(huì)被動(dòng)態(tài)編譯到target目錄下,而且它是個(gè)文件夾,在使用file的時(shí)候是可以從target中讀到我們想要的文件的,但是在打包成jar之后,則file無(wú)法從jar的路徑上獲取文件,這個(gè)時(shí)候就需要使用getResource()和getResourceAsStream()來(lái)動(dòng)態(tài)加載jar中的文件;
這是來(lái)自stackoverflow的回答;
一個(gè)相對(duì)于自己編寫的類路徑,一個(gè)相對(duì)于根路徑進(jìn)行讀取文件.
對(duì)于classloader無(wú)法使用("/")這樣的格式,因?yàn)樗慕^對(duì)路徑是模糊的.
到這里,祝大家過(guò)年好.
爆竹聲中一歲除,春風(fēng)送暖入屠蘇户侥;
千門萬(wàn)戶曈曈日镀琉,總把新桃換舊符。