研究:class.getResource("") VS. class.getClassLoader().getResource("")

笔者最近在一个小测试中遇到了关于getClass().getResource("")和getClass().getClassLoader().getResource("")一个小问题,因为之前对这两个方法理解都不是很深刻,遇到这样的问题总是很抓狂,因此,趁着这次机会好好的研究了一下他们的工作原理,并且记录在这里,方便自己以及有需要的人查阅。

背景

问题的起因是笔者的小测试中引入了部门中的一个第三方包,而这个包中有一行代码跑出了NullPointerException. 具体原因是它使用了getClass().getClassLoader().getResource("")想要获取当前项目的运行目录。而getClass().getClassLoader().getResource("")的结果是null。当笔者遇到这个问题的时候也很懵逼。随即各种尝试,最后使用getClass().getResource("")方法解决了问题。但不明真相。

所以笔者带着两个问题开始进行研究:

1. 为什么getClass().getClassLoader().getResource("")返回的结果是Null

2. getClass().getResource("") 与 getClass().getClassLoader().getResource("") 有什么区别。

Round1

需要注意的是,笔者出问题的小测试是以jar包的形式运行在jvm里的。所以笔者先写了一段小程序。

然后以jar包的形式运行,得到结果如下,和笔者遇到的问题一致。

而当笔者通过在IDE里直接运行下面的程序,结果是OK的。

这里可以发现,貌似使用jar包运行程序与在IDE里面直接运行程序是有差异的。同时也发现,在以jar包的形式运行的时候,getClass().getClassLoader().getResource("") 返回了Null, 而getClass().getResource("") 返回的是一个jar包的路径。

Round2

那么现在可以开始debug了,看一下到底底层做了什么使两种形式返回的结果不一样。首先,需要下载JDK的源码,由于Oracle的JDK只能获取java和javax包下的源码,不全,所以笔者下载了openJDK8的源码。

根据debug的结果,笔者发现当JVM开始运行,会加载classpath下的class文件以及jar包。 在类URLClassPath里有个内部类叫Loader,而这个类就是专门负责加载class文件及jar包的。Loader与ClassLoader的关系可以简单的理解为每一个ClassLoader对应多个Loader。而到底需要多少个Loader来加载文件及jar包是需要根据参数URLs来决定的。用一张图来表示就是


但是Loader它还有两个子类,一个叫FileLoader,另外一个叫JarLoader,不难看出,如果是加载jar文件,那么应该就是用JarLoader来加载了,而如果是class文件那么就应该是用FileLoader来加载了。因而,上面的关系图可以修改为


当调用getClass().getClassLoader().getResource("") 时,其实就是在遍历所有的JarLoader和FileLoader,看有没有相匹配的路径。很明显,笔者遇到的在执行jar文件和IDE里得到不同的结果与JarLoader和FileLoader的实现有直接的关系。

JarLoader是将jar文件加载并将里面的class文件都加载到内存中,形成相对应的一个叫JarEntry的类。所以当执行getClass().getClassLoader().getResource("")时,没有找到与空字符串想匹配的的JarEntry,最终返回null。


FileLoader则不同,它首先知道当前的ClassPath, 所以初始化了一个File(classPath)对象。所以当执行

getClass().getClassLoader().getResource(path)时,最终返回的是File.getPath+"/"+path,既当path为空字符串时,返回的是classPath路径。

到这里,第一个问题基本上解决了。第二个问题网上的回答也比较多,其实从源码可以轻松的找到答案。

getClass().getResource("") 最终也是调ClassLoader.getResource()方法,只不过它多了一步,就是解析传入的name。

当name为空时,返回的就是当前class所在的路径,在笔者的测试环境中就是“com.test”。所以,当调用getClass().getResource("")时就是调用

getClass().getClassLoader().getResource("com/test/")。这也是为什么笔者开始歪打正着的用getClass().getResource("")解决了这个问题。

建议大家再看看源码,动手debug一下,会发现很有意思。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容

  • “定位”这本书其实看了挺久的,初次看的时候没有太大感觉,作者所表达的观点,也是平日工作里已经再操作的思路。 今晚用...
    谌基平阅读 285评论 0 0
  • 被virtual关键字修饰的成员函数就是虚函数。虚函数的作用是实现多态性。所谓多态性就是以共同的方法,对不同的对象...
    意不适阅读 192评论 0 0
  • 夏日的夜晚, 总也会听到一星半点的声音。 譬如青蛙的叫声, 池塘里游鱼甩尾巴的声音, 鸟雀的呓语, 或是晚风的声音...
    南方尾則淑阅读 152评论 0 0
  • 提到“逻辑”一词真的是心里的痛。快大学毕业的时候,考公务员遇到逻辑题一脸懵,答案全靠胡乱猜写。至此以后,对于逻辑好...
    颜洳碧语阅读 2,585评论 0 4