一、Class#getResourceAsStream
1.1 JDK定义
public InputStream getResourceAsStream(String name)
查找具有给定名称的资源。查找与给定类相关的资源的规则,是通过定义类的 class loader 实现的。此方法委托此对象的类加载器。如果此对象通过引导类加载器加载,则此方法将委托给 ClassLoader.getSystemResourceAsStream(java.lang.String)。
在委托前,使用下面的算法从给定的资源名构造一个绝对资源名:
(1)如果 name 以 '/' 开始 ('\u002f'),则绝对资源名是 '/' 后面的 name 的一部分。
(2)否则,绝对名具有以下形式:
modified_package_name/name
其中 modified_package_name 是此对象的
包名,该名用 '/' 取代了 '.' ('\u002e')。
参数:
name - 所需资源的名称
返回:
一个 InputStream 对象;如果找不到带有该名称的资源,则返回 null
抛出:
NullPointerException - 如果 name 是 null
从以下版本开始:
JDK1.1
此方法参数是与getResouce()方法一样的,它相当于你用getResource()取得File文件后,再new InputStream(file)一样的结果。
二、Class#getResource
2.1 JDK定义
[url]public URL getResource(String name)[/url]
查找带有给定名称的资源。查找与给定类相关的资源的规则是通过定义类的 class loader 实现的。此方法委托给此对象的类加载器。如果此对象通过引导类加载器加载,则此方法将委托给 ClassLoader.getSystemResource(java.lang.String)。
在委托前,使用下面的算法从给定的资源名构造一个绝对资源名:
(1)如果 name 以 '/' ('\u002f') 开始,则绝对资源名是 '/'
后面的 name 的一部分。
(2)否则,绝对名具有以下形式:
modified_package_name/name
其中 modified_package_name 是此对象的包名,该名用 '/' 取代了 '.' ('\u002e')。
参数:
name - 所需资源的名称
返回:
一个 URL 对象;如果找不到带有该名称的资源,则返回 null
从以下版本开始:
JDK1.1
2.2 举例说明
包路径:
Ehcache.java的内容:
public static void main(String[] args)
{
// 1输出: /E:/anWorkSpace/pra_test/target/classes/com/anning/cache/
System.out.println(Ehcache.class.getResource("").getFile());
// 2输出: /E:/anWorkSpace/pra_test/target/classes/
System.out.println(Ehcache.class.getResource("/").getFile());
// 3输出: /E:/anWorkSpace/pra_test/target/classes/conf/ehcache.xml System.out.println(Ehcache.class.getResource("/conf/ehcache.xml").getFile());
// 4输出: null
System.out.println(Ehcache.class.getResource("conf/ehcache.xml"));
}
解释:
1.路径开头不带”/”的会当做是相对当前类(class文件)的文件路径,比如第一行输出没有指明任何文件,而且路径开头不带”/”,则取得的是当前class的路径,也就是包路径。
第4行输出则找不到文件,因为不带”/”,所以是相对当前class的相对路径,从图中我们可以看到ehcache.class的包路径下没有conf/ehcache.xml文件。
2.路径开头带”/”的是绝对路径,是相对classpath的路径。如第二行的输出,classpath就是/E:/anWorkSpace/pra_test/target/classes/。
第三行的输出就是相对classpath路径,加上绝对路径的”/”后面的部分。
总结:
可以看到Class的这两个方法都是跟ClassLoader相关的,也就是跟class的路径和classpath路径相关。这与开发工具IDE无关,请看下面:
比如在上面的main方法里写写new File就要写成:
new File(“src/main/resources/conf/ehcache.xml”);
才可以在IDE环境下运行main方法。但是
发布打包之后这又要报错找不到文件,为什么呢?因为IDE里File的相对路径是相对IDE的工程根路径,也就是System.getProperties(“user.dir”)的路径。
new File(path),这个方法的路径到底在那里取决于调用
java命令的起始位置定义在哪里,tomcat/bin下面的catalina.bat调用了java,所以在tomcat下相对起始位置是tomcat/bin,但是eclipse启动时,起始位置是eclipse的项目路径。
三、ClassLoader#getResource
3.1 JDK定义
public URL getResource(String name)
查找具有给定名称的资源。资源是可以通过类代码以与代码路径无关的方式访问的一些数据(图像、声音、文本等)。
资源名称是以 '/' 分隔的标识资源的路径名称。
此方法首先搜索资源的父类加载器;如果父类加载器为 null,则搜索的路径就是虚拟机的内置类加载器的路径。如果搜索失败,则此方法将调用 findResource(String) 来查找资源。
参数:
name - 资源名称
返回:
读取资源的 URL 对象;如果找不到该资源,或者调用者没有足够的权限获取该资源,则返回 null。
从以下版本开始:
1.1
3.2 举例说明
Ehcache.java的内容:
public static void main(String[] args)
{
// 1输出: file:/E:/anWorkSpace/pra_test/target/classes/
System.out.println(Ehcache.class.getClassLoader().getResource(""));
// 2输出: null
System.out.println(Ehcache.class.getClassLoader().getResource("/"));
//3输出:file:/E:/anWorkSpace/pra_test/target/classes/conf/ehcache.xml
System.out.println(Ehcache.class.getClassLoader().getResource("conf/ehcache.xml"));
}
可以看出通过classLoader获取classpath的参数跟Class的不太一样。但原理基本一致,因为Class的方法是调用ClassLoader的对应方法实现的,只是入参做了点点处理。
ClassLoader把决定权交给了类加载器,例如tomcat的类加载是非委托机制的,而weblogic的类加载是委托的。
ClassLoader的getResourceAsStream方法与它的getResource方法的关系跟Class的两个方法是一样的。
还可以参考:
http://blog.csdn.net/fancylovejava/article/details/7577294
http://blog.sina.com.cn/s/blog_9386f17b0100w2vv.html
http://lavasoft.blog.51cto.com/62575/265821/
http://lavasoft.blog.51cto.com/62575/265821/
http://www.iteye.com/topic/483115
最后一篇提到,可以使用XWork的工具类ClassLoaderUtil读取jar包里的资源文件等。
- 大小: 6.2 KB
分享到:
相关推荐
JDK1.8源码分析 相关的原始码分析结果会以注解的形式体现到原始码中 ... ClassLoader#getResource Class#getResource Throwable ServiceLoader SPI 参考引用类型 ThreadLocal 线程 Proxy ClassLoader
classloader 加密解密应用程序 ,反编译class
NULL 博文链接:https://ldbjakyo.iteye.com/blog/1046984
让Java支持热加载是个不错的想法。如何做到的呢? 1. 定义好接口和实现类 2. 让代理类通过反射的方式调用实现类,对外暴露的是代理类。 3. 自定义URLClassLoader。检查实现类.class文件的...Java自定义classloader;
Java 虚拟机中ClassLoader 相关简介 双亲委托机制 Android 中ClassLoader 简介
ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的
破解java加密的ClassLoader.java,在classloader植入破解代码
ClassLoader原理,ClassLoader原理 ClassLoader原理
自定义classloader的使用
NULL 博文链接:https://lz12366.iteye.com/blog/735289
Java ClassLoader定制实例
自定义ClassLoader,控制台输入调试。 运行期间 重新载入指定目录的class文件。可实现对于类的功能函数更新。 用到java 反射,@interface 等技术
一个开源的Cplusplus类加载器,基于它实现了一个简单的例子,见我写的classloader的文章。
破解java加密的rt.jar,在classloader植入破解代码,默认输出到c:/TEMP/classes/目录。使用方法:只要下载本rt.jar,然后替换掉jdk1.8.0_25\jre\lib目录下的rt.jar。然后运行你需要破解的java程序即可,如果你的java...
java classloader classpath 张孝祥
理解Java ClassLoader机制
Classloader
用于验证理解Android中Classloader加载类机制的程序demo,从中可以对比DexClassLoader和PathClassLoader的区别联系。
eclipse工程格式 博文链接:https://aga.iteye.com/blog/200818
并且只有调用了newInstance()方法采用调用构造函数,创建类的对象看下Class.forName()源码//Class.forName(String c