先來看下A和B兩個(gè)模塊
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 ??
問題來了
單獨(dú)運(yùn)行A模塊,或單獨(dú)運(yùn)行B模塊都是正常的,但將A和B兩個(gè)模塊集成后運(yùn)行,Spring容器就啟動不了了:
到底出了啥問題
隨便搜索了一下,還發(fā)現(xiàn)很多人遇到這個(gè)問題,這個(gè)就是來自stackoverflow的問題:
http://stackoverflow.com/questions/7940452/spring-application-context-not-able-to-load-property-placeholder-properties
可惜啊,好像都沒有人給出正確的解決。
那究竟是什么問題呢?也想了很久哦....終于回想起來了(寫書時(shí)讀過Spring源碼),原來是Spring容器采用反射掃描的發(fā)現(xiàn)機(jī)制,在探 測到Spring容器中有一個(gè) org.springframework.beans.factory.config.PropertyPlaceholderConfigurer的 Bean就會停止對剩余PropertyPlaceholderConfigurer的掃描(Spring 3.1已經(jīng)使用PropertySourcesPlaceholderConfigurer替代 PropertyPlaceholderConfigurer了)。
而<context:property-placeholder/>這個(gè)基于命名空間的配置,其實(shí)內(nèi)部就是創(chuàng)建一個(gè)PropertyPlaceholderConfigurer Bean而已。
換句話說,即Spring容器僅允許最多定義一個(gè)PropertyPlaceholderConfigurer(或<context:property-placeholder/>),其余的會被Spring忽略掉
(其實(shí)Spring如果提供一個(gè)警告就好了)。
拿上來的例子來說,如果A和B模塊是單獨(dú)運(yùn)行的,由于Spring容器都只有一個(gè)PropertyPlaceholderConfigurer, 因此屬性文件會被正常加載并替換掉。如果A和B兩模塊集成后運(yùn)行,Spring容器中就有兩個(gè) PropertyPlaceholderConfigurer Bean了,這時(shí)就看誰先誰后了, 先的保留,后的忽略!因此,只加載到了一個(gè)屬性文件,因而造成無法正確進(jìn)行屬性替換的問題。
咋解決呢?
定位問題需要9999元錢,解決問題只需要1元錢
。
屬性文件加載在統(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ì)想想是有道理的,一個(gè)項(xiàng)目或一個(gè)系統(tǒng)的配置應(yīng)該放在一起,不宜分散。
這樣才可以做到統(tǒng)一管控,否則到處都有配置,到底是加載哪個(gè)配置文件呢?有時(shí)你還會不小心讓JAR中的Spring配置文件加載一個(gè)位于JAR中 的屬性文件,而外面有更改不了。如果Spring使用了這種機(jī)制,即使JAR包中的Spring配置文件使用<context:property- placeholder/>引用到JAR中的屬性文件,只要你要外而的Spring配置文件中顯示提供一 個(gè)<context:property-placeholder/>指定另一個(gè)屬性文件 ,就可以覆蓋JAR中的默認(rèn)配置了。
想了一想,Spring這樣做是利大于弊的。
?
注意:如果有父子容器,如web應(yīng)用,則應(yīng)該各自配置一個(gè)屬性文件,這樣不會有問題,最終結(jié)論,每個(gè)spring容器只能有一個(gè) PropertyPlaceholderConfigurer。
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯(lián)系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長非常感激您!手機(jī)微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
