Spring 3.1 –从数据库加载XML配置的属性

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/370653.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何判断PHP 是线程安全还是非线程安全的

什么是线程安全与非线程安全&#xff1f; 线程安全就是在多线程环境下也不会出现数据不一致&#xff0c;而非线程安全就有可能出现数据不一致的情况。 线程安全由于要确保数据的一致性&#xff0c;所以对资源的读写进行了控制&#xff0c;换句话说增加了系统开销。所以在单线程…

关联查询mysql_《MySQL数据库》关联查询

原标题&#xff1a;《MySQL数据库》关联查询一、关联查询1、概念在查询数据时&#xff0c;所需要的数据不只在一张表中&#xff0c;可能在两张或多张表中。这个时候&#xff0c;需要同时操作这些表来查询数据&#xff0c;即关联查询。关联查询所涉及到的表与表之间都会存在有关…

python语言语块句的标记_《自然语言处理理论与实战》

编辑推荐 1.讲解自然语言处理的理论 2.案例丰富&#xff0c;实战性强 3.适合自然语言处理学习的入门者 内容提要 自然语言处理是什么&#xff1f;谁需要学习自然语言处理&#xff1f;自然语言处理在哪些地方应用&#xff1f;相关问题一直困扰着不少初学者。针对这一情况&#x…

NOIP2017年11月9日赛前模拟

最后一次NOIP模拟了 题目1&#xff1a;回文数字 Tom 最近在研究回文数字。  假设 s[i] 是长度为 i 的回文数个数&#xff08;不含前导0&#xff09;&#xff0c;则对于给定的正整数 n 有&#xff1a; 以上等式中最后面的括号是布尔表达式&#xff0c;Tom 想知道S[n] mod 2333…

height百分比失效

heigh:100%失效 解决方案&#xff1a; 第一种 html, body { height: 100%; } 第二种 div { height: 100%; position: absolute; } 非定位元素的宽高百分比计算不会将 padding 计算在内&#xff0c;而定位元素会计算在内。 利用这个特性可以实现图片左右半区点击分别上一张图…

Java堆空间,本机堆和内存问题

最近&#xff0c;我在和一个朋友讨论为什么Java进程使用的内存比启动Java进程时设置的最大堆多。 代码创建的所有Java对象都是在Java堆空间内创建的&#xff0c;其大小由-Xmx选项定义。 但是一个Java进程由很多空间组成&#xff0c;而不仅仅是Java堆空间。 以下是组成Java进程…

mysql视图表怎么设置约束_MySQL一一sql的视图、索引、约束

一、视图本质上相当于一张**“虚拟表”**&#xff0c;可当作独立的一张表进行操作(增、删、改、查)** 作用&#xff1a;**** a)**可通过权限控制&#xff0c;只将“表中的少数列”暴露给数据库用户&#xff0c;而不让该用户直接操纵数据库中“实际表”** b)**…

Software Development Life Cycle

转载于:https://www.cnblogs.com/genezhao/p/6879848.html

python中 的用法_详解python中@的用法

python中的用法 是一个装饰器&#xff0c;针对函数&#xff0c;起调用传参的作用。 有修饰和被修饰的区别&#xff0c;function作为一个装饰器&#xff0c;用来修饰紧跟着的函数&#xff08;可以是另一个装饰器&#xff0c;也可以是函数定义&#xff09;。 代码1 结果1 Its fun…

ArrayAndString(数组和字符串)

1.实现一个算法&#xff0c;确定一个字符串的所有字符是否全都不同。假使不允许使用额外的数据结构&#xff0c;又该怎么处理&#xff1f; public class UniqueChars {public static void main(String[] args) {// TODO Auto-generated method stubString string "abcdef…

MyBatis教程– CRUD操作和映射关系–第2部分

为了说明这一点&#xff0c;我们正在考虑以下示例域模型&#xff1a; 会有用户&#xff0c;每个用户可能都有一个博客&#xff0c;每个博客可以包含零个或多个帖子。 这三个表的数据库结构如下&#xff1a; CREATE TABLE user (user_id int(10) unsigned NOT NULL auto_incr…

position 的属性值

理论上来说&#xff0c;全部 position 的取值有8个 包括&#xff1a;position&#xff1a;static | relative | absolute | fixed | sticky | initial | inherit | unset 其中最常用的是 static 、relative、absolute、fixed 和 sticky initial、inherit、unset 是css的关键…

[ JavaScript ] JavaScript 实现继承.

对于javascript中的继承&#xff0c;因为js中没有后端语言中的类式继承。所以js中的继承&#xff0c;一般都是原型继承(prototype)。 function P (name){this.name name;this.say function(){console.log(p);} }function S (name,id){this.id id;this.eat function(){conso…

mysql数据库应用的权限层级_MySQL数据库的用户权限管理

嗨&#xff01;各位小伙伴今天翻了一下历史记录MySQL 数据库还有点内容今天开始我们就来补上吧~用户权限管理伙伴们要知道&#xff0c;在数据库方面有两个方向。一个是数据库管理员(Database Administrator)简称DBA&#xff0c;一个是数据库开发工程师(Database Developer)&…

linux i2c adapter 增加设备_Linux驱动之I2C驱动架构

一、Linux的I2C体系结构主要由三部分组成&#xff1a;(1) I2C核心提供I2C控制器和设备驱动的注册和注销方法&#xff0c;I2C通信方法&#xff0c;与适配器无关的代码以及探测设备等。(2) I2C控制器驱动(适配器)(3) I2C设备驱动二、重要的结构体i2c_adapter//i2c控制器(适配器)i…

Alpha-end

前言 失心疯病源10团队代码管理github个人感悟 肝不动了&#xff0c;肝不动了。明天如果见不到我&#xff0c;不要太想我。站立会议 队名&#xff1a;PMS530雨勤&#xff08;组长&#xff09; 今天完成了那些任务 熬夜肝代码代码签入github明天的计划 肝到凌晨还剩下哪些任务 团…

html 01前沿-web介绍

1. 认识网页 网页主要由文字、图像和超链接等元素构成。当然&#xff0c;除了这些元素&#xff0c;网页中还可以包含音频、视频以及Flash等。 2. 浏览器&#xff08;显示代码&#xff09; 浏览器是网页显示、运行的平台&#xff0c;常用的浏览器有IE、火狐&#xff08;Firefox…

避免写慢SQL

最近在整理数据库中的慢SQL&#xff0c;同时也查询了相关资料。记录一下&#xff0c;要学会使用执行计划来分析SQL。 1. 为查询缓存优化你的查询 大多数的MySQL服务器都开启了查询缓存。这是提高性最有效的方法之一&#xff0c;而且这是被MySQL的数据库引擎处理的。当有很多相同…

为什么子孙后代会讨厌使用java.util.Stack

在我用无意义的重言式杀死你之前&#xff0c;这是要点 如果您的应用程序接近实时&#xff0c;或者将代码发送到Mars&#xff0c;则需要保留Java中默认的Stack实现。 根据LinkedList编写您自己的版本。 同样&#xff0c;如果您的应用程序是关键任务&#xff0c;并且希望堆栈由…

play 连接mysql_Play framework 2.x 连接mysql | 学步园

笔者所使用的系统为64位 windows7。本文假设java1.5版本以上环境已经搭好&#xff0c;play 框架已经下载至本地。首先我们创建一个项目。命令行进入play的目录命令&#xff1a;play new demo再次输入项目名字输入2 选择java项目创建完成界面OK&#xff0c;一个play框架下的java…