关于jar读取文件路径问题

 

关于jar读取文件路径的这个问题可愁死我了!

(以下内容如有不准确,或错误的地方请留言.我会积极配合领导去查证并更正我的错误.)

先看项目树:(对应项目Open in Terminal    ---   tree >>E:\tree.txt)

E:.
└─src
    └─main
        ├─java
        │  └─com
        │      └─ycc
        │          └─netty
        │              ├─bean
        │              ├─constant
        │              └─util
        └─resources

当我想把项目打成jar包并发布到服务器上的时候,它总是这样亲切的对我说:

An exception java.lang.IllegalArgumentException: bound must be positive

我苦思冥想,然后.......我就去google了.......

最终以这种方式解决了问题:

    NameUtil.class.getClassLoader().getResourceAsStream("surname.json");

究其原因,其实是这个样子(...)

打包之后,jar包中的所加载的文件路径发生了变化,我们在把 ~.*(例如:test.json)打包到C盘之后,其路径变为file:/C:/*.jar!/~.*,如果你在原项目中使用new File(filePath)之类的方法来加载的话,肯定会找不到资源文件。主要是因为Jar包是一个单独的文件而非文件夹,绝对不可能通过file:/C:/.../*.jar!/~.*这种形式的文件URL来定位~.*。所以即使是相对路径,也无法定位到Jar包内的资源文件。

这个时候我们需要使用到getResource()getResourceAsStream();

这两个方法没有区别,只是返回的值不一样.


首先上一段测试代码:

            System.out.println(NameUtil.class.getClassLoader());
            System.out.println(NameUtil.class.getResource("/surname.json"));
            System.out.println(NameUtil.class.getResource("./surname.json"));
            System.out.println(NameUtil.class.getResource("surname.json"));


            System.out.println(NameUtil.class.getClassLoader().getResource("/surname.json"));
            System.out.println(NameUtil.class.getClassLoader().getResource("./surname.json"));
            System.out.println(NameUtil.class.getClassLoader().getResource("surname.json"));

它的结果是这样的(surname.json是放在resources下面的,打包会直接打到classes的根目录下);

file:/E:/YCC/nettyServer/common/target/classes/surname.json
null
null
null
file:/E:/YCC/nettyServer/common/target/classes/surname.json
file:/E:/YCC/nettyServer/common/target/classes/surname.json

我们发现这样一个现象:

NameUtil.class.getResource("surname.json");

NameUtil.class.getClassLoader().getResource("surname.json");

结果是不一样的....一脸懵逼...

先试着脑补2分钟.


当我们使用NameUtil.class.getResource("surname.json")的时候Class.java解析出来的路径是:

它的解析代码如下:

    /**
     * Add a package name prefix if the name is not absolute Remove leading "/"
     * if name is absolute
     */
private String resolveName(String name) {
        if (name == null) {
            return name;
        }
        if (!name.startsWith("/")) {
            Class<?> c = this;
            while (c.isArray()) {
                c = c.getComponentType();
            }
            String baseName = c.getName();
            int index = baseName.lastIndexOf('.');
            if (index != -1) {
                name = baseName.substring(0, index).replace('.', '/')
                    +"/"+name;
            }
        } else {
            name = name.substring(1);
        }
        return name;
    }

其实就是如果路径是以"/"开头的,则指绝对路径(相对于项目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中的文件;

Class.getResource can take a "relative" resource name, which is treated relative to the class's package. Alternatively you can specify an "absolute" resource name by using a leading slash. Classloader resource paths are always deemed to be absolute.

So the following are basically equivalent:

foo.bar.Baz.class.getResource("xyz.txt");
foo.bar.Baz.class.getClassLoader().getResource("foo/bar/xyz.txt");

And so are these (but they're different from the above):

foo.bar.Baz.class.getResource("/data/xyz.txt");
foo.bar.Baz.class.getClassLoader().getResource("data/xyz.txt");

这是来自stackoverflow的回答;

一个相对于自己编写的类路径,一个相对于根路径进行读取文件.

对于classloader无法使用("/")这样的格式,因为它的绝对路径是模糊的.

到这里,祝大家过年好.


爆竹声中一岁除,春风送暖入屠苏;
千门万户曈曈日,总把新桃换旧符。

 

春节放爆竹

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值