先來看下A和B兩個模塊
A模塊和B模塊都分別擁有自己的Spring XML配置售貌,并分別擁有自己的配置文件:
A模塊
A模塊的Spring配置文件如下:
<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd">
<context:property-placeholder location="classpath*:conf/conf_a.properties"/>
<bean class="com.xxx.aaa.Bean1"
p:driverClassName="${modulea.jdbc.driverClassName}"
p:url="${modulea.jdbc.url}"
p:username="${modulea.jdbc.username}"
p:password="${modulea.jdbc.password}"/>
</beans>
其配置文件位于類路徑conf/conf_a.properties中:
modulea.jdbc.driverClassName=com.mysql.jdbc.Driver
modulea.jdbc.username=cartan
modulea.jdbc.password=superman
modulea.jdbc.url=jdbc:mysql://127.0.0.1:3306/modulea?useUnicode=true&characterEncoding=utf8
B模塊
B模塊的Spring配置文件如下:
<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd">
<context:property-placeholder location="classpath*:conf/conf_b.properties"/>
<bean class="com.xxx.bbb.Bean1"
p:driverClassName="${moduleb.jdbc.driverClassName}"
p:url="${moduleb.jdbc.url}"
p:username="${moduleb.jdbc.username}"
p:password="${moduleb.jdbc.password}"/>
</beans>
其配置文件位于類路徑conf/conf_b.properties中:
moduleb.jdbc.driverClassName=com.mysql.jdbc.Driver
moduleb.jdbc.username=cartan
moduleb.jdbc.password=superman
moduleb.jdbc.url=jdbc:mysql://127.0.0.1:3306/modulea?useUnicode=true&characterEncoding=utf8
問題來了
單獨運行A模塊漆弄,或單獨運行B模塊都是正常的,但將A和B兩個模塊集成后運行,Spring容器就啟動不了了:
Could not resolve placeholder 'moduleb.jdbc.driverClassName' in string value "${moduleb.jdbc.driverClassName}"[/quote]
到底出了啥問題
隨便搜索了一下,還發(fā)現(xiàn)很多人遇到這個問題,這個就是來自stackoverflow的問題:
http://stackoverflow.com/questions/7940452/spring-application-context-not-able-to-load-property-placeholder-properties
可惜啊,好像都沒有人給出正確的解決梦谜。
那究竟是什么問題呢?也想了很久哦....終于回想起來了(寫書時讀過Spring源碼)袭景,原來是Spring容器采用反射掃描的發(fā)現(xiàn)機(jī)制唁桩,在探測到Spring容器中有一個org.springframework.beans.factory.config.PropertyPlaceholderConfigurer的Bean就會停止對剩余PropertyPlaceholderConfigurer的掃描(Spring 3.1已經(jīng)使用PropertySourcesPlaceholderConfigurer替代PropertyPlaceholderConfigurer了)。
而<context:property-placeholder/>
這個基于命名空間的配置耸棒,其實內(nèi)部就是創(chuàng)建一個PropertyPlaceholderConfigurer Bean而已朵夏。換句話說,即Spring容器僅允許最多定義一個PropertyPlaceholderConfigurer(或<context:property-placeholder/>)榆纽,其余的會被Spring忽略掉(其實Spring如果提供一個警告就好了)仰猖。
拿上來的例子來說,如果A和B模塊是單獨運行的奈籽,由于Spring容器都只有一個PropertyPlaceholderConfigurer饥侵,因此屬性文件會被正常加載并替換掉。如果A和B兩模塊集成后運行衣屏,Spring容器中就有兩個PropertyPlaceholderConfigurer Bean了躏升,這時就看誰先誰后了, 先的保留狼忱,后的忽略膨疏!因此,只加載到了一個屬性文件钻弄,因而造成無法正確進(jìn)行屬性替換的問題佃却。
咋解決呢?
定位問題需要9999元錢窘俺,解決問題只需要1元錢 :D 饲帅。
屬性文件加載在統(tǒng)一的地方做,不要分模塊加載即可瘤泪。
A模塊a.xml:
<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd">
<!--<context:property-placeholder location="classpath*:conf/conf_a.properties"/>-->
<bean class="com.xxx.aaa.Bean1"
p:driverClassName="${modulea.jdbc.driverClassName}"
p:url="${modulea.jdbc.url}"
p:username="${modulea.jdbc.username}"
p:password="${modulea.jdbc.password}"/>
</beans>
B模塊b.xml:
<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd">
<!--<context:property-placeholder location="classpath*:conf/conf_b.properties"/>-->
<bean class="com.xxx.bbb.Bean1"
p:driverClassName="${moduleb.jdbc.driverClassName}"
p:url="${moduleb.jdbc.url}"
p:username="${moduleb.jdbc.username}"
p:password="${moduleb.jdbc.password}"/>
</beans>
集成:
<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd">
<context:property-placeholder location="classpath*:conf/conf*.properties"/>
<import resource="a.xml"/>
<import resource="b.xml"/>
</beans>
進(jìn)一步思考
為什么霸畋谩?Spring為什么要這樣呢对途?細(xì)想想是有道理的赦邻,一個項目或一個系統(tǒng)的配置應(yīng)該放在一起,不宜分散实檀。
這樣才可以做到統(tǒng)一管控惶洲,否則到處都有配置按声,到底是加載哪個配置文件呢?有時你還會不小心讓JAR中的Spring配置文件加載一個位于JAR中的屬性文件湃鹊,而外面有更改不了儒喊。如果Spring使用了這種機(jī)制镣奋,即使JAR包中的Spring配置文件使用<context:property-placeholder/>引用到JAR中的屬性文件币呵,只要你要外而的Spring配置文件中顯示提供一個<context:property-placeholder/>指定另一個屬性文件 ,就可以覆蓋JAR中的默認(rèn)配置了侨颈。
想了一想余赢,Spring這樣做是利大于弊的。