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