最近在看spring的资源获取时发现JDK里存在几种不同方式的资源获取,因比较混乱特地总结起来帮助和我一样混乱的人理解。下面是我项目的类结构图,在 src/main/java 下有两个类 ResourceTest.java和Resource.javaresources 目录下有两个资源文件 request.xml 和 conf/sysConf.json

├── pom.xml
├── src
│ ├── main
│ │ ├── java
│ │ │ ├── com
│ │ │ │ ├── alipay
│ │ │ │ │ ├── ResourceTest.java
│ │ │ │ │ └── Resource.java
│ │ └── resources
│ │ │ ├── conf
│ │ │ │ ├── sysConf.json
│ │ │ └── request.xml
└── local.iml

在ResourceTest中,我想获取Resource这个类以及request.xml、sysConf这两个资源文件,可以分为Class和ClassLoader两种方式来获取资源,而ClassLoader则又可以细分为3种方式:

public class ResourceTest {

    public static void main(String[] args) {
// 1、通过Class的getResource方法
String a1 = ResourceTest.class.getResource("/com/alipay/Resource.class").getPath();
String a2 = ResourceTest.class.getResource("Resource.class").getPath();
String a3 = ResourceTest.class.getResource("/request.xml").getPath();
String a4 = ResourceTest.class.getResource("../../request.xml").getPath();
String a5 = ResourceTest.class.getResource("/conf/sysConf.json").getPath();
String a6 = ResourceTest.class.getResource("../../conf/sysConf.json").getPath(); // 2、通过本类的ClassLoader的getResource方法
String b1 = ResourceTest.class.getClassLoader().getResource("com/alipay/Resource.class").getPath();
String b2 = ResourceTest.class.getClassLoader().getResource("request.xml").getPath();
String b3 = ResourceTest.class.getClassLoader().getResource("conf/sysConf.json").getPath(); // 3、通过ClassLoader的getSystemResource方法
String c1 = ClassLoader.getSystemClassLoader().getResource("com/alipay/Resource.class").getPath();
String c2 = ClassLoader.getSystemClassLoader().getResource("request.xml").getPath();
String c3 = ClassLoader.getSystemClassLoader().getResource("conf/sysConf.json").getPath(); // 4、通过ClassLoader的getSystemResource方法
String d1 = ClassLoader.getSystemResource("com/alipay/Resource.class").getPath();
String d2 = ClassLoader.getSystemResource("request.xml").getPath();
String d3 = ClassLoader.getSystemResource("conf/sysConf.json").getPath(); // 5、通过Thread方式
String e1 = Thread.currentThread().getContextClassLoader().getResource("com/alipay/Resource.class").getPath();
String e2 = Thread.currentThread().getContextClassLoader().getResource("request.xml").getPath();
String e3 = Thread.currentThread().getContextClassLoader().getResource("conf/sysConf.json").getPath();
}
}

以上所有的方式都能够获取到对应的资源文件。

由于maven打包会把 src/main/javasrc/main/resources 下的文件放到 target/classes 下,所以下面统一以根路径代表此目录,总结起来有以下几个规律:

  • Class.getResource()的资源获取如果以 / 开头,则从根路径开始搜索资源。
  • Class.getResource()的资源获取如果不以 / 开头,则从当前类所在的路径开始搜索资源。
  • ClassLoader.getResource()的资源获取不能以 / 开头,统一从根路径开始搜索资源。

下面还是老习惯,翻开源码看看为什么是这样的规律。

Class.getResource()

public java.net.URL getResource(String name) {
name = resolveName(name);
// 获得类的类加载器,默认为AppClassLoader
ClassLoader cl = getClassLoader0();
if (cl==null) {
// A system class.
return ClassLoader.getSystemResource(name);
}
return cl.getResource(name);
}

可以很清晰的看出上面的资源获取流程:

  1. 解析文件路径,变成ClassLoader所支持的路径。
  2. 获取该类的类加载器,默认为AppClassLoader,接着调用它的getResource方法。
  3. 如果类加载器获取失败,直接走ClassLoader的getSystemResource方法来获取

我们看看resolveName的解析规则:

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();
// 截取当前类所在的包和name使用 / 进行拼接
int index = baseName.lastIndexOf('.');
if (index != -1) {
name = baseName.substring(0, index).replace('.', '/')
+"/"+name;
}
} else {
// 如果以 / 开头,截取 / 后面的内容
name = name.substring(1);
}
return name;
}

原理和我们上面分析的一样。因为Class的getResource最终还是调用的ClassLoader,所以我们接着来看ClassLoader的相关资源获取方法。

ClassLoader.getResource()

public URL getResource(String name) {
URL url;
if (parent != null) {
// 递归调用
url = parent.getResource(name);
} else {
// 使用BootstrapClassLoader发现资源
url = getBootstrapResource(name);
}
if (url == null) {
// 真正去找对应的url
url = findResource(name);
}
return url;
}

这个方法比较有意思的地方在于它使用了双亲委派机制来加载资源(回顾双亲委派机制 点我 ),它从BootstrapClassLoader一层层往下找直到最后找到该资源。本例中是通过AppClassLoader来找到了对应的资源(实际使用了URLClassLoader的findResource方法)

ClassLoader.getSystemResource()

这种方式对应于例子中的方式四,相比于方式三就多了一个空判断。

public static URL getSystemResource(String name) {
// systemClassLoader就是从Launcher获取的AppClassLoader
ClassLoader system = getSystemClassLoader();
if (system == null) {
return getBootstrapResource(name);
}
return system.getResource(name);
}

getSystemResource和getResource的区别就在于你是否实现了自己的类加载器,如果都是使用的默认的AppClassLoader,这两个方法的作用一样。

线程上下文加载方式

这种加载方式对应于例子中的最后一种方式,它是使用 java.lang.Thread 中的方法 getContextClassLoader()setContextClassLoader(ClassLoader cl)用来获取和设置线程的上下文类加载器。如果没有通过 setContextClassLoader(ClassLoader cl)方法进行设置的话,线程将继承其父线程的上下文类加载器。Java 应用运行的初始线程的上下文类加载器是 AppClassLoader ,具体设置参考 sun.misc.Launcher 的构造函数。

更多关于线程上下文加载的疑问可以参考之前我的一篇文章:理解TCCL:线程上下文类加载器

因为线程上下文加载方式的灵活性,所以推荐在资源文件的读取时使用。

转自 http://benjaminwhx.com/2018/07/12/%E8%AF%B4%E8%AF%B4Java%E4%B8%AD%E7%9A%84%E8%B5%84%E6%BA%90%E6%96%87%E4%BB%B6%E7%9A%84%E8%AF%BB%E5%8F%96/

说说Java中的资源文件的读取的更多相关文章

  1. java 中获得 资源文件方法

    1 java 中获取资源文件的方法 项目目录如下 获取当前项目的目录路径 方法一:使用类名 MergeDocHandler.class.getClassLoader().getResource(&qu ...

  2. java基础知识3--如何获取资源文件(Java中获取资源文件的url)

    java开发中,常见的resource文件有:.xml,.properties,.txt文件等,后台开发中经常用到读取资源文件,处理业务逻辑,然后返回结果. 获取资源文件的方法说明getResourc ...

  3. Java中的资源文件加载方式

    文件加载方式有两种: 使用文件系统自带的路径机制,一个应用程序只能有一个当前目录,但可以有Path变量来访问多个目录 使用ClassPath路径机制,类路径跟Path全局变量一样也是有多个值 在Jav ...

  4. Java中获取资源文件的方法总结

    这里总结3中方法获取资源文件的 ServletContext Class ClassLoader 文件的位置 1. ServletContext public void doGet(HttpServl ...

  5. java中io创建文件和读取文件

    简单了解IO流:https://www.cnblogs.com/weibanggang/p/10034325.html package com.wbg.iodemo1128; import java. ...

  6. Jar中的Java程序如何读取Jar包中的资源文件

    Jar中的Java程序如何读取Jar包中的资源文件 比如项目的组织结构如下(以idea中的项目为例): |-ProjectName |-.idea/  //这个目录是idea中项目的属性文件夹 |-s ...

  7. java 资源文件的读取

    Java将配置文件当作一种资源(resource)来处理,并且提供了两个类来读取这些资源,一个是Class类,另一个是ClassLoader类. gradle 项目 项目目录结构  用Class类加载 ...

  8. 读取web工程中.properties资源文件的模板代码

    读取web工程中.properties资源文件的模板代码 // 读取web工程中.properties资源文件的模板代码 private void test2() throws IOException ...

  9. Java加载资源文件的两种方法

    处理配置文件对于Java程序员来说再常见不过了,不管是Servlet,Spring,抑或是Structs,都需要与配置文件打交道.Java将配置文件当作一种资源(resource)来处理,并且提供了两 ...

随机推荐

  1. POI创建Excel使用的常见的属性

    public static void main(String[] args) { //创建新的Excel 工作簿 HSSFWorkbook workbook =new HSSFWorkbook(); ...

  2. HTML5中表单验证的8种方法

    HTML5中表单验证的8种方法 2012-4-21 11:00| 发布者: benben| 查看: 2765| 评论: 0 摘要: 前一篇,我们介绍了HTML5中新的表单特性和函数, 今天就继续来谈谈 ...

  3. 【Linux】Shell三类变量的作用域——linux shell “永久环境变量”、“临时环境变量”和"普通变量"之完全解读

      2015-05-08 00:15 3896人阅读 评论(10) 收藏 举报 本文章已收录于:   分类: 软件开发进阶(419) 作者同类文章X Unix/Linux杂项(118) 作者同类文章X ...

  4. 【Javascript】js图形编辑器库介绍

    10个JavaScript库绘制自己的图表 jopen 2015-04-06 18:18:38 • 发布 摘要:10个JavaScript库绘制自己的图表 JointJS JointJS is a J ...

  5. Java读写二进制文件示例

    相对于文本文件,二进制文件读写快,定位快而准,下面是代码示例: import java.io.DataInput; import java.io.DataOutput; import java.io. ...

  6. javascript脚本中使用json2.js解析json

    官方地址:https://github.com/douglascrockford/JSON-js   点击页面右下角“Download ZIP”下载   网页中引用json2.js,下面是一个简单的例 ...

  7. vue - 条件语句

    1.与小程序不同之处一,小程序无论变量还是常亮都可以用双向绑定来解决{{}},而vue一旦双(单)引号包起来以后就失效了. 2.注意一点,切记双引号注意不要混淆哈,这里是一排双引号包单引号,那里是一排 ...

  8. C++ STL中Map的按Key排序

    为了实现快速查找,map内部本身就是按序存储的(比如红黑树).在我们插入<key, value>键值对时,就会按照key的大小顺序进行存储.这也是作为key的类型必须能够进行<运算比 ...

  9. 3、jQuery的DOM基础

    DOM模型在页面文档中,通过树状模型展示页面的元素和内容,其展示的方式则是通过节点(node)来实现的. 3.1 访问元素 3.1.1 元素属性操作 Attr()可以对元素属性执行获取和设置操作,而r ...

  10. ubuntu使用du命令查看一级子目录存储空间大小

    命令如下: ls | xargs du -ksh 可以ls不同的目录以查看不同的目录下的一级子目录大小.直接使用ls为当前目录下的一级子目录大小. 查看其他目录的大小: ls -d dirname/* ...