在使用 jar 执行 java 代码时,有一个需求是从 jar 包所在目录的同级目录下读取配置文件的需求,从网上找了很多方法感觉都挺复杂的,在这里总结一下.
>以classpath
开头的 URL
表示该文件为jar
包内文件的路径.
如:classpath://config/app.config
表示jar
包根路径config文件夹下的app.config文件
以file
开头的URL
表示该文件为jar
包外文件的路径
如:file://./config/app.config
表示
摘要
// 当前我想从jar包的同级目录下读取一个名为 'config.txt'的文件的话,我需要指定目录为.
File file = new File(".","config.txt")
说明
File file = new File("config.txt")
当只包含文件名称时,java程序会默认尝试从jar包的根路径去读取文件,当尝试使用 file.getCanonicalPath()
方法读取时,便会得到该文件在jar包内的路径.
示例
我当前的工程的路径为D:\WorkSpace\path_demo01\
在工程执行以下java代码:
当指定parent
时:
- 会从
parent
下查找path
资源.
log(FileUtil.file(".", "/config/app.config").getCanonicalPath());
log(FileUtil.file(".", "config/app.config").getCanonicalPath());
// D:\WorkSpace\path_demo01\config\app.config
//加载与当前jar包同级目录下的文件
log(FileUtil.file(".", "app.config").getCanonicalPath());
当没有指定parent
:
- 如果
path
为绝对路径
时,会从绝对路径
下查找 - 如果
path
为相对路径
时,会从classpath
的根路径下开始查找
log(FileUtil.file("/config/app.config").getCanonicalPath());
// D:\config\app.config
log(FileUtil.file("config/app.config").getCanonicalPath());
// D:\WorkSpace\path_demo01\target\classes\config\app.config
通过当前类加载资源:
- 如果
path
为相对路径
会指定要加载的资源路径与当前类所在包的路径一致 - 如果
path
为绝对路径
,那么就会从classpath
的根路径下开始查找
log(App.class.getResource("/config/app.config"));
// file:/D:/WorkSpace/path_demo01/target/classes/config/app.config
log(App.class.getResource("config/app.config"));
// file:/D:/WorkSpace/path_demo01/target/classes/top/ghimi/config/app.config
通过类加载器加载资源:
默认是从ClassPath
根下获取,path
不能以/
开头,最终是由ClassLoader
获取资源.
log(App.class.getClassLoader().getResource("/config/app.config"));
// null
log(App.class.getClassLoader().getResource("config/app.config"));
// file:/D:/WorkSpace/path_demo01/target/classes/config/app.config
加载jar
包内的资源
当代码打包成jar包的形式后,是无法通过new File()
的形式加载jar
包内的资源的.此时有可能抛出
FileNotFoundException
异常,是由于将path
当成jar包外的目录查找不到资源导致的.URI is not hierarchical
异常,是由于无法直接读取jar
包中资源(透明)而抛出的异常.
解决方法:
使用 getResourceAsStream()
方法直接获取资源的流
而不是getResource()
获取资源文件对象的方式读取资源.
//修改前,未打包成jar包时能够正常执行,打包后会抛出异常
log(App.class.getClassLoader().getResource("config/app.config"));
// 修改后,打成jar包后也可以正常加载资源
log(App.class.getClassLoader().getResourceAsStream("config/app.config"));
加载jar
包外的资源
- 会从
parent
目录下查找path
资源.
//加载与当前jar包同级目录下的文件
log(FileUtil.file(".", "app.config").getCanonicalPath());
// D:\WorkSpace\path_demo01\app.config
//加载与当前jar包的上一级目录下的文件
log(FileUtil.file("..", "app.config").getCanonicalPath());
// D:\WorkSpace\app.config
借用工具hutool
:
<dependency>
<groupid>cn.hutool</groupid>
<artifactid>hutool-all</artifactid>
<version>4.5.6</version>
</dependency>
参考资料:
来源:oschina
链接:https://my.oschina.net/ghimi/blog/3171473