Spring裏PropertyPlaceholderConfigurer類的使用

1. PropertyPlaceholderConfigurer是個bean工廠後置處理器的實現,也就是 BeanFactoryPostProcessor接口的一個實現。PropertyPlaceholderConfigurer能夠將上下文(配置文 件)中的屬性值放在另外一個單獨的標準java Properties文件中去。在XML文件中用${key}替換指定的properties文件中的值。這樣的話,只須要對properties文件進 行修改,而不用對xml配置文件進行修改。html

2.在Spring中,使用PropertyPlaceholderConfigurer能夠在XML配置文件中加入外部屬性文件,固然也能夠指定外部文件的編碼,如:java

複製代碼
<bean id="propertyConfigurer"class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">

   <property name="location">

     <value>conf/sqlmap/jdbc.properties</value>

   </property>

    <property name="fileEncoding">

      <value>UTF-8</value>

    </property>

</bean>
複製代碼

 

固然也能夠引入多個屬性文件,如:mysql

複製代碼
<bean id="propertyConfigurer"class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">

  <property name="locations">

   <list>

    <value>/WEB-INF/mail.properties</value>  

    <value>classpath: conf/sqlmap/jdbc.properties</value>//注意這兩種value值的寫法

   </list>

  </property>

</bean>
複製代碼

 

3.譬如,jdbc.properties的內容爲:spring

複製代碼
jdbc.driverClassName=com.mysql.jdbc.Driver

jdbc.url=jdbc:mysql://localhost/mysqldb?useUnicode=true&amp;characterEncoding=UTF-8&amp;zeroDateTimeBehavior=round;

jdbc.username=root

jdbc.password=123456
複製代碼

 

4.那麼在spring配置文件中,咱們就能夠這樣寫:sql

複製代碼
<bean id="propertyConfigurer"class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">

  <property name="locations">

   <list>

    <value>classpath: conf/sqlmap/jdbc.properties </value>

   </list>

  </property>

</bean>

<bean id="dataSource"class="org.apache.commons.dbcp.BasicDataSource"destroy-method="close">

  <property name="driverClassName"value="${jdbc.driverClassName}" />

  <property name="url" value="${jdbc.url}" />

  <property name="username" value="${jdbc.username}"/>

  <property name="password"value="${jdbc.password}" />

</bean>
複製代碼

 

5.這樣,一個簡單的數據源就設置完畢了。能夠看出:PropertyPlaceholderConfigurer起的做用就是將佔位符指向的數據庫配置信息放在bean中定義的工具。數據庫

6.查看源代碼,能夠發現,locations屬性定義在PropertyPlaceholderConfigurer的祖父類 PropertiesLoaderSupport中,而location只有 setter方法。相似於這樣的配置,在spring的源程序中很常見的。apache

PropertyPlaceholderConfigurer若是在指定的Properties文件中找不到你想使用的屬性,它還會在Java的System類屬性中查找。工具

咱們能夠經過System.setProperty(key, value)或者java中經過-Dnamevalue來給Spring配置文件傳遞參數。編碼

 

爲簡化PropertyPlaceholderConfigurer的使用,Spring提供了<context:property-placeholder/>元素。下面給出了配置示例,啓用它後,開發者便不用配置PropertyPlaceholderConfigurer對象了。url

  1. <context:property-placeholder location="userinfo.properties"/> 

  PropertyPlaceholderConfigurer內置的功能很是豐富,若是它未找到${xxx}中定義的xxx鍵,它還會去JVM系統屬性(System.getProperty())和環境變量  (System.getenv())中尋找。經過啓用systemPropertiesMode和searchSystemEnvironment屬性,開發者可以控制這一行爲。

這個在spring中配置文件中是很是經常使用的。

context:property-placeholder大大的方便了咱們數據庫的配置。

 

[html]  view plain  copy
 
  1. 只須要在spring的配置文件裏添加一句:<context:property-placeholder?location="classpath:jdbc.properties"/>?便可,這裏location值爲參數配置文件的位置,參數配置文件一般放在src目錄下,而參數配置文件的格式跟java通用的參數配置文件相同,即鍵值對的形式,例如:  
  2.   
  3. #jdbc配置  
  4.   
  5. test.jdbc.driverClassName=com.mysql.jdbc.Driver  
  6. test.jdbc.url=jdbc:mysql://localhost:3306/test  
  7. test.jdbc.username=root  
  8. test.jdbc.password=root  


這樣一來就能夠爲spring配置的bean的屬性設置值了,好比spring有一個jdbc數據源的類DriverManagerDataSource

 

在配置文件裏這麼定義bean:

<bean id="testDataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="${test.jdbc.driverClassName}"/>
<property name="url" value="${test.jdbc.url}"/>
<property name="username" value="${test.jdbc.username}"/>
<property name="password" value="${test.jdbc.password}"/>
</bean>

這樣修改起來也方便,也統一的這個規範。

 

另外須要注意的是,若是遇到下面着着這種問題:

A模塊和B模塊都分別擁有本身的Spring XML配置,並分別擁有本身的配置文件:

A模塊的Spring配置文件以下:

[html]  view plain  copy
 
  1. <?xml version="1.0" encoding="UTF-8" ?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  4.        xmlns:context="http://www.springframework.org/schema/context"  
  5.        xmlns:p="http://www.springframework.org/schema/p"  
  6.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd  
  7.        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd">  
  8.    <context:property-placeholder location="classpath*:conf/conf_a.properties"/>  
  9.    <bean class="com.xxx.aaa.Bean1"  
  10.           p:driverClassName="${modulea.jdbc.driverClassName}"  
  11.           p:url="${modulea.jdbc.url}"  
  12.           p:username="${modulea.jdbc.username}"  
  13.           p:password="${modulea.jdbc.password}"/>  
  14. </beans>  
conf/conf_a.properties:
[html]  view plain  copy
 
  1. modulea.jdbc.driverClassName=com.mysql.jdbc.Driver  
  2. modulea.jdbc.username=cartan  
  3. modulea.jdbc.password=superman  
  4. modulea.jdbc.url=jdbc:mysql://127.0.0.1:3306/modulea?useUnicode=true&characterEncoding=utf8  


B模塊的Spring配置文件以下:
[html]  view plain  copy
 
  1. <?xml version="1.0" encoding="UTF-8" ?>    
  2. <beans xmlns="http://www.springframework.org/schema/beans"    
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"    
  4.        xmlns:context="http://www.springframework.org/schema/context"    
  5.        xmlns:p="http://www.springframework.org/schema/p"    
  6.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd    
  7.        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd">    
  8.    <context:property-placeholder location="classpath*:conf/conf_b.properties"/>    
  9.    <bean class="com.xxx.bbb.Bean1"    
  10.           p:driverClassName="${moduleb.jdbc.driverClassName}"    
  11.           p:url="${moduleb.jdbc.url}"    
  12.           p:username="${moduleb.jdbc.username}"    
  13.           p:password="${moduleb.jdbc.password}"/>    
  14. </beans>    
conf/conf_b.properties:
[html]  view plain  copy
 
  1. moduleb.jdbc.driverClassName=com.mysql.jdbc.Driver  
  2. moduleb.jdbc.username=cartan  
  3. moduleb.jdbc.password=superman  
  4. 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}"
緣由:Spring容器採用反射掃描的發現機制,在探測到Spring容器中有一個org.springframework.beans.factory.config.PropertyPlaceholderConfigurer的Bean就會中止對剩餘PropertyPlaceholderConfigurer的掃描

spring容器中最多隻能定義一個context:property-placeholder,否則就出現那種個錯誤,那如何來解決上面的問題呢?

 

A和B模塊去掉

[html]  view plain  copy
 
  1. <context:property-placeholder location="classpath*:conf/conf_b.properties"/>   

 

而後從新寫個xml:

 

[html]  view plain  copy
 
  1. <?xml version="1.0" encoding="UTF-8" ?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  4.        xmlns:context="http://www.springframework.org/schema/context"  
  5.        xmlns:p="http://www.springframework.org/schema/p"  
  6.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd  
  7.        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd">  
  8.    <context:property-placeholder location="classpath*:conf/conf*.properties"/>  
  9.    <import resource="a.xml"/>  
  10.    <import resource="b.xml"/>  
  11. </beans>  
相關文章
相關標籤/搜索