在应用Spring的工程中,使用class path的方式加载配置文件应该是最常用的做法,然而对大部分人来说,刚开始使用Spring时,几乎都碰到过加载配置文件失败的情况,除了配置上的错误外,很多时候是因为配置文件的路径和程序中指定的加载路径不一致,从而导致配置文件找不到,或是加载了错误地方的配置文件。本文将就Spring如何从class path中加载配置文件做一些简要的分析。
情形一:使用classpath加载且不含通配符 这是最简单的情形,Spring默认会使用当前线程的ClassLoader的
1.当工程目录结构如图所示: 即配置文件放在bin目录中的conf文件夹里,这时使用
Loading XML bean definitions from class path resource [conf/application-context.xml]
2.
当工程目录结构如图所示:
即
bin
目录下只有
.class
文件,没有配置文件,同时在工程属性的
Java Build Path->Libraries
里导入
conf.jar
文件,
jar
文件结构如图所示:
这时使用
Loading XML bean definitions from class path resource [conf/application-context.xml]
3.
当工程目录结构如图所示
:
即配置文件放在
bin
目录中的
conf
文件夹里,同时在工程属性的
Java Build Path->Libraries
里导入
conf.jar
文件,
jar
文件结构如图所示:
这时使用
Loading XML bean definitions from
class path resource [conf/application-context.xml]
情形二:使用classpath加载,包含通配符 碰到通配符的情况时,Spring会通过使用路径中的非通配符部分先确定资源的大致位置,然后根据这个位置在确定具体的资源位置,结合下面给出的几种情况可以更好地理解Spring的这种工作方式
1. 当工程目录结构如图所示:
即配置文件放在bin目录中的conf文件夹里,这时使用
来创建ApplicationContext对象的话,Spring首先会通过路径中的非通配符部分即conf,先确定conf的路径,即bin/conf目录,然后从该目录下加载配置文件,由于使用了/**/的方式,表明要加载conf目录下包括各级子目录中的所有配置文件,因此bin/conf/application-context.xml文件和 bin/conf/admin/admin-application-context.xml都会被加载,Spring启动时的输出显示为: Loading XML bean definitions from file [D:\myworkspace\spring-study\bin\conf\admin\admin-application-context.xml] Loading XML bean definitions from file [D:\myworkspace\spring-study\bin\conf\application-context.xml]
2
.当工程目录结构如图所示:
即
bin
目录下只有
.class
文件,没有配置文件,同时在工程属性的
Java Build Path->Libraries
里导入
conf.jar
文件,
jar
文件结构如图所示:
这时使用
conf/admin/admin-application-context.xml都会被加载,Spring启动时的输出显示为: Loading XML bean definitions from class path resource [conf/admin/admin-application-context.xml] Loading XML bean definitions from class path resource [conf/application-context.xml]
3
.当工程目录结构如图所示:
即配置文件放在
bin
目录中的
conf
文件夹里,同时在工程属性的
Java Build Path->Libraries
里导入
conf.jar
文件,
jar
文件结构如图所示:
这时使用
bin/conf/admin/admin-application-context.xml都会被加载,但conf.jar文件中的配置文件并不会被加载,Spring启动时的输出显示为: Loading XML bean definitions from file [D:\myworkspace\spring-study\bin\conf\admin\admin-application-context.xml] Loading XML bean definitions from file
[D:\myworkspace\spring-study\bin\conf\application-context.xml]
情形三:使用classpath*前缀且不包含通配符 使用classpath*前缀可以获取所有与给定路径匹配的classpath资源,从而避免出现两个不同位置有相同名字的文件,Spring只加载其中一个的情况。
当工程目录结构如图所示
:
即配置文件放在
bin
目录中的
conf
文件夹里,同时在工程属性的
Java Build Path->Libraries
里导入
conf.jar
文件,
jar
文件结构如图所示:
这时使用
Loading XML bean definitions from URL [file:/D:/myworkspace/spring-study/bin/conf/application-context.xml] Loading XML bean definitions from URL [jar:file:/D:/myworkspace/conf1.jar!/conf/application-context.xml] 来创建 ApplicationContext 对象的话, Spring 将会加载 bin 目录下的 application-context.xml 文件和 jar 包里的 application-context.xml 文件, Spring 启动时的输出显示为: |
情形四:使用classpath*前缀,包含通配符 当工程目录结构如图所示:
即配置文件放在
bin
目录中的
conf
文件夹里,同时在工程属性的
Java Build Path->Libraries
里导入
conf.jar
文件,
jar
文件结构如图所示:
这时使用
bin/conf/admin/admin-application-context.xml以及jar包中的 conf/application-context.xml和 conf/admin/admin-application-context.xml都会被加载,Spring启动时的输出显示为: Loading XML bean definitions from file [D:\myworkspace\spring-study\bin\conf\admin\admin-application-context.xml] Loading XML bean definitions from file [D:\myworkspace\spring-study\bin\conf\application-context.xml] Loading XML bean definitions from URL [jar:file:/D:/myworkspace/conf1.jar!/conf/admin/admin-application-context.xml] Loading XML bean definitions from URL [jar:file:/D:/myworkspace/conf1.jar!/conf/application-context.xml]
特别注意:
如果工程目录如图所示:
即配置文件直接放在
bin
目录中,同时在工程属性的
Java Build Path->Libraries
里导入
conf.jar
文件,
jar
文件结构如图所示:
这时使用
bin/application-context.xml和bin/admin/admin-application-context.xml, 而jar包中的配置文件并不会被加载。这是因为Spring使用class path加载配置文件时需要借助JDK的ClassLoader.getResources(String name)方法,而该方法有一个局限:当传入的参数为空字符串时,即我们本意是想从根目录获取文件,这时JDK只会返回存在于文件系统中的资源,而在jar包中的资源并不会被返回。
我们在eclipse中写个简单的测试类就能很容易看到这点:
运行测试类后,输出结果为: file:/D:/myworkspace/spring-study/bin/ file:/D:/ProgramFiles/eclipse3.2/configuration/org.eclipse.osgi/bundles/47/1/.cp/
可以看到,获得的资源路径中并不包含
jar
包中的路径,因此
jar
包中的配置文件自然不能被
Spring
加载了。
来创建
ApplicationContext
对象的话,
Spring
只会加载
|