`

spring tool configure

阅读更多

If you use Spring in your Java projects, you are probably familiar with the PropertyPlaceholderConfigurer. Using this BeanFactoryPostProcessor, you can pull property values from a properties file. An example XML context snippet from the Spring JavaDocs:

<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
    <property name="driverClassName"><value>${driver}</value></property>
    <property name="url"><value>jdbc:${dbname}</value></property>
</bean>

And the corresponding properties file:

driver=com.mysql.jdbc.Driver
dbname=mysql:mydb

Such a technique separates configuration from the context file. You could simply use a different properties file to configure development, QA, staging, and production environments.

What, however, if you would like to use the same technique to pull property values from a database table? It's actually relatively easy to achieve. First, you override the default PropertyPlaceholderConfigurer:

import ...;
 
/**
 * Used to load configurtion properties from a database table. 
 * Changes the default placeholderPrefix to '#{'.
 * 
 * &lt;property name="url" value="#{website.url}"/&gt;
 * 
 * @author David Norton
 *
 */
public class DatabasePropertyPlaceholderConfigurer extends PropertyPlaceholderConfigurer {
    public DatabasePropertyPlaceholderConfigurer() {
        super();
        setPlaceholderPrefix("#{");
    }
}

Then, you add some database code and override the loadProperties() function:

private JdbcTemplate jdbcTemplate;
 
@Override
protected void loadProperties(final Properties props) throws IOException {
    jdbcTemplate.query("SELECT "+nameColumn+", "+valueColumn+" FROM "+propsTable,new RowCallbackHandler() {
 
        public void processRow(ResultSet rs) throws SQLException {
            String name = rs.getString(nameColumn);
            String value = rs.getString(valueColumn);
            props.setProperty(name, value);
        }
 
    });
}
 
public void setDataSource(DataSource dataSource) {
    this.jdbcTemplate = new JdbcTemplate(dataSource);
}
 
private String nameColumn = "name";
private String valueColumn = "value";
private String propsTable = "config";

In order to use this, place the following in your Spring XML (assuming you already have a configured datasource bean named 'dataSource'):

<bean id="propertyConfigurer" class="DatabasePropertyPlaceholderConfigurer">
    <property name="dataSource" ref="dataSource"/>
</bean>
 
<bean id="someBean" class="org.my.ConfigurableService">
    <property name="website.url" value="#{website.url}"/>
</bean>

Note the departure from the traditional "${propertyname}" usage to "#{propertyname}".

We can add some setters to make the table and column names configurable:

/**
 * Specify a custom column for the property name. Default is 'name'.
 * @param nameColumn
 */
public void setNameColumn(String nameColumn) {
    this.nameColumn = nameColumn;
}
/**
 * Specify a custom column for the property value. Default is 'value'.
 * @param valueColumn
 */
public void setValueColumn(String valueColumn) {
    this.valueColumn = valueColumn;
}
 
/**
 * Specify a custom table name. Default is 'config'.
 * @param propsTable
 */
public void setPropsTable(String propsTable) {
    this.propsTable = propsTable;
}

Used such as:

<bean id="propertyConfigurer" class="DatabasePropertyPlaceholderConfigurer">
    <property name="dataSource" ref="dataSource"/>
    <property name="nameColumn" ref="property_name"/>
</bean>

Some future refactorings could include: move the actually data access code out of this class and into a more generic properties data access object. Of course, comments and criticism are welcome.

About David

<< Just check out my LinkedIn profile. You can also catch posts at my company's corporate blog, The TPS Report. http://threepillarsoftware.com/blog
Comments (1) Trackbacks (0)
  1. Hi

    Great Post,
    I tried to use your solution in my project but i bumped into a one big inconvenience.
    Next to DatabasePropertyPlaceholderConfigurer i wanted to use “standard”
    PropertyPlaceholderConfigurer to resolve properties on dataSource bean.
    These properties must be loaded from property file.

    bean id=”dataSource”
    class=”org.springframework.jdbc.datasource.DriverManagerDataSource”
    property name=”driverClassName” value=”${db.driver}”"
    property name=”url” value=”${url}”
    property name=”username” value=”${db.username}”
    property name=”password” value=”${db.password}”
    bean

    The problem is that Spring first creates ALL PropertyPlaceholderConfigurer beans and after that it invokes their postProcessBeanFactory() methods.
    In this way dataSource bean is created before any placeholder resolving takes place.

    To cope with it i decided to inject dataSource bean name instead of dataSource bean itself.
    Additionally, let DatabasePropertyPlaceholderConfigurer implements BeanFactoryAware
    (PropertyPlaceholderConfigurer implements BeanFactoryAware but it stores BeanFactory in private filed and can not access it from subclasses)
    and retrieve dataSource (by “dataSourceBeanName” name from BeanFactory) within loadProperties method.
    This way i am able to retrieve datasource bean “lazily” after standard PropertyPlaceholderConfigurer resolving took place.

分享到:
评论

相关推荐

    Mastering Spring Cloud-Packt Publishing(2018)

    You will learn to configure the Spring Cloud server and run the Eureka server to enable service registration and discovery. Then you will learn about techniques related to load balancing and circuit ...

    spring-boot-reference.pdf

    Spring Boot Documentation 1. About the Documentation 2. Getting Help 3. First Steps 4. Working with Spring Boot 5. Learning about Spring Boot Features 6. Moving to Production 7. Advanced Topics II. ...

    spring-security3 入门篇

    - **Spring Security Tool Kit (SST)**: 提供了一些辅助工具,简化Spring Security的配置和使用。 - **Spring Security Test**: 提供了测试支持,方便进行安全相关的单元测试和集成测试。 6. **进阶话题** - **...

    Spring Security3 配置使用

    在开发过程中,利用IDE的插件(如Spring Tool Suite)可以方便地进行配置和调试。 总结,Spring Security 3的配置使用涵盖了从基础的认证和授权,到复杂的源码理解和工具集成。理解并掌握这些知识点,将有助于构建...

    简单实现Spring Quartz定时器

    至于工具,像`qconf`(Quartz Configuration and Management Tool)这样的图形化工具可以帮助我们更直观地管理和监控Quartz调度器。此外,IDE插件如Eclipse的Quartz plugin也提供了方便的集成开发环境支持。 总的来...

    Ubuntu 18 搭建openjdk源码 报错误 configure: error: Could not find required tool for ZIPEXE

    前几天看完了深入理解 Java 虚拟机-JVM 高级特性与最佳实践(第2版) 这本书突发奇想想研究一下jdk的源码,...SpringCloud-基础设施即服务day3-Docker上 拷贝这段代码进入虚拟机(ps:注意文件路径) docker run --rm -i

    Spring Boot打包war jar 部署tomcat

    -- https://docs.spring.io/spring-boot/docs/current/reference/html/build-tool-plugins-maven-plugin.html --&gt; ``` 在实际开发中,我们可以根据需要选择合适的方法来打包和部署 Spring Boot 应用程序。如果您...

    lombok.jar在eclipse和sts中做相应配置可以代替get、set

    Spring Tool Suite (STS)则是基于Eclipse构建的一款专门针对Spring框架及其生态系统的开发工具。两者都支持Java开发,并且可以通过安装插件来增强功能,比如支持Lombok。 #### 安装Lombok插件 1. **下载Lombok** ...

    camel-manual-2.0

    There are multiple ways to add routes and configure Camel, including using Spring or directly in Java, as shown in the example above. #### Architecture The architecture of Apache Camel is centered ...

    idea jenkins集成及Hybris构建小技巧

    2. **配置Jenkins连接**:重启后,在Tools Windows中找到Jenkins按钮(若未显示,可通过View -&gt; Tool Windows找到)。在Jenkins窗口中配置服务器地址,包括Crumb Data。Crumb Data是Jenkins的安全机制,防止跨站请求...

    金蝶BOSV6.1_业务组件API参考手册

    Packages com.kingdee.bos ...com.kingdee.bos.engine.difftool com.kingdee.bos.engine.impl com.kingdee.bos.framework com.kingdee.bos.framework.agent com.kingdee.bos.framework.batch ...

    MySQL 5.6 Reference Manual

    Table of Contents Preface, Notes, Licenses . . . . . . . . ....1. Licenses for Third-Party Components ....1.1. FindGTest.cmake License ....1.2. LPeg Library License ....1.3. LuaFileSystem Library License ....

Global site tag (gtag.js) - Google Analytics