下午做spa安装程序的时候,由于在jar文件中包含了config.properties文件,导致花了近两个小时来解决"uri is not hierarchical"问题,真是郁闷。就因为一点小疏忽……
不过,对JVM ClassLoader加载体系有了更加清楚的理解。
JVM的类加载体系,相信读过《Inside JVM》的人可能比较熟悉一些。在1.2之后,类加载器体系就是双亲委派体系。依次从高到低级排序是:
启动类加载器, 标准类加载器, 系统类加载器(也就是类路径加载器,JVM实现的用户类加载器),用户自定义类加载器。
我们的资源一般都是通过系统类加载器来进行加载的,比如class文件,jar文件,以及其它资源。
在eclipse项目中,属性文件config.properties直接放在src根目录下即可,在程序中,直接:
java 代码
- URL url = this.getClass().getClassLoader().getSystemResource("config.properties");
- URI cfgUri = URI.create(url.toExternalForm());
- File cfgFile = new File(cfgUri);
这样就OK了。它自然会在classpath上找到这个config.properties。
那如果把这个文件放在JAR文件中,做为JAR的资源呢,其实也是没有关系,上面的代码也是可以找到JAR文件中的资源的,一点错误都没有。同时,也可以这么来加载:
java 代码
- URL url = Thread.currentThread().getContextClassLoader().getResource(fileName);
一样,都是从JAR文件中去加载文件。
但是,如果jar文件中包含了config.properties,并且jar文件外面也包含了config.properties文件,那么结果会怎么样呢?
注意,下面用====包围的解释是不正确的。我自己弄错了双亲委派,哈哈。详细说明:
http://java.sun.com/docs/books/tutorial/ext/basics/load.html
=============================================================================================
因为classloader加载是如果自己加载不了,才让它的双亲classLoader来加载。
这里,由于加载jar的classloader,暂且叫做jarloader可以在jar文件中找到config.properties文件,那么显然就不用去麻烦它的双亲去加载了。要达到加载外面config.properties文件的效果,必须在jar文件中将它去除,免得jarloader麻烦它的双亲classloader去加载它,因为jarloader是被系统类加载器加载的,因此,系统类加载器在classpath上继续查找,找到config.properties文件,这样就加载成功了。
=============================================================================================
应该是class path classloader加载资源顺序的问题。