Spring使通過其PropertyPlaceholderConfigurer和(Spring 3.1之前)PropertySourcesPlaceholderConfigurer(Spring 3.1)從屬性文件中獲取的值易于注入。 這些類實現了BeanFactoryPostProcessor接口,該接口使它們能夠在初始化bean之前在Spring XML配置文件中操作值。 因此,如果您指定將$ {jdbc.driverClassName}設置為屬性“ driverClassName”,則該變量將被替換/交換為屬性文件中帶有鍵“ jdbc.driverClassName”的值。
除了屬性文件之外,數據庫表還可以是獲取鍵值對的地方。 太好了,所以只需擴展PropertySourcesPlaceholderConfigurer,并讓它讀取包含鍵值對的表,然后填充它們就可以了!
但是,有一個小問題。 如果DataSource bean也依賴于從屬性文件獲得的值(例如JDBC URL,用戶名,密碼)并且是出色的Springer,請將此bean注入擴展PropertySourcesPlaceholderConfigurer的bean類中,則bean容器將無法正確啟動,因為' jdbc.driverClassName'變量無法解析。 奇怪,但事實如此。
這樣做的原因是,任何注入到BeanFactoryPostProcessor類中的bean都會在BeanFactoryPostProcessor類運行之前觸發Bean初始化。 您知道,依賴注入…所有依賴的bean必須先準備好才能注入到使用者中。 因此,這創建了一種循環依賴的東西。 在運行BeanFactoryPostProcessor類之前,首先要解析XML配置中的所有依賴關系。
那么,如何處理呢? 好吧,您可以使用一個技巧。 BeanFactoryPostProcessor類可以通過“ postProcessBeanFactory”方法訪問ConfigurableListableBeanFactory對象。 從該對象,您可以執行“ getBean”并獲取具有ID的任何bean的引用。 猜猜是什么,您可以獲取吹噓的DataSource bean,而無需觸發過早的bean初始化。
假設有一個包含以下數據的表“ sys_param”:
PARAM_CD PARAM_VALUE
-------------- --------------
service.charge 1.5
rebate.amount 15.99
smtp.ip 173.194.79.16
DbPropertySourcesPlaceholderConfigurer如下所示:
package org.gizmo.labs.utils.spring;import javax.sql.DataSource;import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.context.support.PropertySourcesPlaceholderConfigurer;public class DbPropertySourcesPlaceholderConfigurer extends PropertySourcesPlaceholderConfigurer
{@Overridepublic void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException{DataSource dataSource = beanFactory.getBean(DataSource.class);DbProperties dbProps = new DbProperties(dataSource);setProperties(dbProps);super.postProcessBeanFactory(beanFactory);}
}
DbProperties類將使用DataSource引用并查詢數據庫以獲取鍵值對:
package org.gizmo.labs.utils.spring;import java.util.List;
import java.util.Map;
import java.util.Properties;import javax.sql.DataSource;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.jdbc.core.JdbcTemplate;public class DbProperties extends Properties
{private final Logger logger = LoggerFactory.getLogger(DbProperties.class);private static final long serialVersionUID = 1L;public DbProperties(DataSource dataSource){super();JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource); List<map> l = jdbcTemplate.queryForList('select param_cd, param_value from sys_param');for(Mapm: l){logger.debug('Loading from DB: [{}:{}]', m.get('PARAM_CD'), m.get('PARAM_VALUE'));setProperty((m.get('PARAM_CD')).toString(), (m.get('PARAM_VALUE')).toString());}}
}
為了證明表中的值已正確注入,下面是充當使用者的類:
package org.gizmo.labs.utils.spring;import java.math.BigDecimal;import org.apache.commons.lang.builder.ReflectionToStringBuilder;
import org.apache.commons.lang.builder.ToStringStyle;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.InitializingBean;public class DbPropConsumer implements InitializingBean
{private final Logger logger = LoggerFactory.getLogger(DbPropConsumer.class);private BigDecimal serviceCharge;private double rebateAmount;private String smtpIp;@Overridepublic void afterPropertiesSet() throws Exception{logger.debug('I have consumed: {}', this);}public String toString(){return ReflectionToStringBuilder.toString(this, ToStringStyle.MULTI_LINE_STYLE);} public BigDecimal getServiceCharge() {return serviceCharge;}public void setServiceCharge(BigDecimal serviceCharge) {this.serviceCharge = serviceCharge;}public double getRebateAmount() {return rebateAmount;}public void setRebateAmount(double rebateAmount) {this.rebateAmount = rebateAmount;}public String getSmtpIp() {return smtpIp;}public void setSmtpIp(String smtpIp) {this.smtpIp = smtpIp;}}
最后但并非最不重要的一點是,Spring配置(未顯示DataSource bean,為清楚起見進行了簡化):
classpath:system.properties
前兩個bean定義是BeanFactoryPostProcessor類,并且為了確保第一個被首先運行,設置了'order'屬性(值越低優先級越高)。
對于DbPropertySourcesPlaceholderConfigurer,為了清楚起見,使用了不同的占位符前綴和后綴(請注意DbPropConsumer的占位符)。
因此,在Spring容器啟動時,您應該能夠查看類似的輸出,如下所示:
2012-09-18 00:03:14, DEBUG, org.gizmo.labs.utils.spring.DbProperties, Loading from DB: [service.charge:1.5]
2012-09-18 00:03:14, DEBUG, org.gizmo.labs.utils.spring.DbProperties, Loading from DB: [rebate.amount:15.99]
2012-09-18 00:03:14, DEBUG, org.gizmo.labs.utils.spring.DbProperties, Loading from DB: [smtp.ip:173.194.79.16]
2012-09-18 00:03:14, DEBUG, org.gizmo.labs.utils.spring.DbPropConsumer, I have consumed: org.gizmo.labs.utils.spring.DbPropConsumer@189b939[
logger=Logger[org.gizmo.labs.utils.spring.DbPropConsumer]
serviceCharge=1.5
rebateAmount=15.99
smtpIp=173.194.79.16
]
參考: Spring 3.1 – YK的Workshop博客中的JCG合作伙伴 Allen Julia 從數據庫加載XML配置的屬性 。
翻譯自: https://www.javacodegeeks.com/2012/11/spring-3-1-loading-properties-for-xml-configuration-from-database.html