Class.getResource(String path)
path不以'/'开头时,默认是今后类所在的包下取资源;path以'/'开头时,则是从项目的ClassPath根下获取资源。在这里'/'表示ClassPath
JDK设置这样的规则,是很好理解的,path不以'/'开头时,咱们就能获取与当前类所在的路径相同的资源文件,而以'/'开头时能够获取ClassPath根下任意路径的资源。
以下所示的例子:java
运行结果为:
file:/D:/work_space/java/bin/net/swiftlet/
file:/D:/work_space/java/bin/swift
Class.getClassLoader().getResource(String path)
path不能以'/'开头时,path是指类加载器的加载范围,在资源加载的过程当中,使用的逐级向上委托的形式加载的,'/'表示Boot ClassLoader中的加载范围,由于这个类加载器是C++实现的,因此加载范围为null。以下所示:jvm
运行结果为:
file:/D:/work_space/java/bin/
null
从上面能够看出:
class.getResource("/") == class.getClassLoader().getResource("")
其实,Class.getResource和ClassLoader.getResource本质上是同样的,都是使用ClassLoader.getResource加载资源的。下面请看一下jdk的Class源码:this
从上面就能够看才出来:Class.getResource和ClassLoader.getResource本质上是同样的,而且Class.getResource的实现时考虑类加载器为null的状况,直接使用更方便。至于为何Class.getResource(String path)中path能够'/'开头,是由于在name = resolveName(name);进行了处理:spa
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
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
(
)
;
int
index
=
baseName
.
lastIndexOf
(
'.'
)
;
if
(
index
!=
-
1
)
{
name
=
baseName
.
substring
(
0
,
index
)
.
replace
(
'.'
,
'/'
)
+
"/"
+
name
;
}
}
else
{
//若是是以"/"开头,则去掉
name
=
name
.
substring
(
1
)
;
}
return
name
;
}
|