关于jar读取文件路径的这个问题可愁死我了!
(以下内容如有不准确,或错误的地方请留言.我会积极配合领导去查证并更正我的错误.)
先看项目树:(对应项目Open in Terminal --- tree >>E:\tree.txt)
当我想把项目打成jar包并发布到服务器上的时候,它总是这样亲切的对我说:
我苦思冥想,然后.......我就去google了.......
最终以这种方式解决了问题:
究其原因,其实是这个样子(...)
打包之后,jar包中的所加载的文件路径发生了变化,我们在把 ~.*(例如:test.json)打包到C盘之后,其路径变为file:/C:/*.jar!/~.*,如果你在原项目中使用new File(filePath)之类的方法来加载的话,肯定会找不到资源文件。主要是因为Jar包是一个单独的文件而非文件夹,绝对不可能通过file:/C:/.../*.jar!/~.*这种形式的文件URL来定位~.*。所以即使是相对路径,也无法定位到Jar包内的资源文件。
这个时候我们需要使用到getResource()和getResourceAsStream();
这两个方法没有区别,只是返回的值不一样.
首先上一段测试代码:
它的结果是这样的(surname.json是放在resources下面的,打包会直接打到classes的根目录下);
我们发现这样一个现象:
NameUtil.class.getResource("surname.json");
和
NameUtil.class.getClassLoader().getResource("surname.json");
结果是不一样的....一脸懵逼...
先试着脑补2分钟.
当我们使用NameUtil.class.getResource("surname.json")的时候Class.java解析出来的路径是:
它的解析代码如下:
其实就是如果路径是以"/"开头的,则指绝对路径(相对于项目classes的绝对路径!),如果不是........则
是获取了当前类NameUtil(因为是NameUtil.class.getResource(),所有这里获取的是NameUtil的命名空间)的类路径.然后根据这个路径来作为参照物.对传入的path进行参照解析.如果你是绝对则没有参照价值,如果是相对路径,则传入的path就必须与NameUtil的类在同一个包下才可以正确解析.
我们再来看 NameUtil.class.getClassLoader().getResource("surname.json");
System.out.println(NameUtil.class.getClassLoader());
它的值是:sun.misc.Launcher$AppClassLoader@18b4aac2
也就是说NameUtil的类加载器是AppClassLoader,
(关于类加载器这里有一篇文章,将的很详细! java中的ClassLoader详解)
AppClassLoader是classes的默认加载器,即咱们的项目的class都是由这家伙加载的.所以它的路径就是classes的根路径,但是它没有对"/"做处理...它的绝对路径是模糊的,这里对于"/"只能返回空值.针对于这方面的问题,以后再研究.这里默认得到NameUtil.class.getClassLoader().getResource的相对路径是根路径.为了验证这点
System.out.println(NameUtil.class.getClassLoader().getResource("surname.json"));
得到的是这样的路径
这里说一下 "./"和"/"不一样,它代表当前路径.
NameUtil.class.getClassLoader().getResource("surname.json")得到的路径和
NameUtil.class.getResource("/surname.json")是一样的,所以这里得到一个等式:
NameUtil.class.getClassLoader().getResource("surname.json")==NameUtil.class.getResource("/surname.json");
到这里,我们读取文件的路径问题基本算是解决了.
总结:
在我们的IDE中,编写好的java代码会被动态编译到target目录下,而且它是个文件夹,在使用file的时候是可以从target中读到我们想要的文件的,但是在打包成jar之后,则file无法从jar的路径上获取文件,这个时候就需要使用getResource()和getResourceAsStream()来动态加载jar中的文件;
这是来自stackoverflow的回答;
一个相对于自己编写的类路径,一个相对于根路径进行读取文件.
对于classloader无法使用("/")这样的格式,因为它的绝对路径是模糊的.
到这里,祝大家过年好.
爆竹声中一岁除,春风送暖入屠苏;
千门万户曈曈日,总把新桃换旧符。