Spring 3.0中的多个属性文件

时间:2012-01-19 11:00:10

标签: java spring properties

我正在开发一个包含多个独立模块的项目,每个模块都有自己的应用程序上下文属性文件。我希望能够加载所有这些属性,以便它们可以用于Spring的占位符解析。

之前的问题已经提到了这一点,并且有一篇很好的博客文章here,它描述了如何在每个上下文中使用PropertyPlaceholderConfigurer,按优先级排序并将ignoreUnresolveablePlaceholders设置为true,以便这些属性文件可以交叉引用每个其他没有炸毁。

然而,这并没有解决我的问题,因为我也希望能够使用我正在加载的属性来进行一些自定义占位符解析(来自我解析的一些yaml文件)。这需要使用PropertyPlaceholderHelper,它需要将Properties对象作为参数。

据我所知,潜在的解决方案是:

1)将所有属性文件合并到一个属性bean中。然后可以使用它来创建PropertyPlaceholderConfigurer(用于Spring的内部占位符解析)并与PropertyPlaceholderHelper一起使用(用于我自己的占位符解析)

2)以某种方式配置PropertyPlaceholderHelper以使用PropertyPlaceholderConfigurers持有的属性集及其分层排列,如果我继续并遵循该博客帖子的建议。

不幸的是我无法弄清楚如何做其中任何一个。任何帮助将不胜感激!

PS看起来Spring 3.1在这方面会有很大的帮助......遗憾的是我们还没有准备好转向它,所以我仍然需要一个解决方案让我度过难关!

****编辑****

感谢目前为止的答案。他们是很好的答案,但遗憾的是我不会帮助我(因为我们之前没有提到这一点而道歉)我们目前正在将我们项目的核心模块与非核心模块分开。这意味着核心模块及其应用程序上下文无法对属性文件的名称进行硬编码。令人沮丧的是,Spring的类路径扫描似乎被打破了,所以“classpath *:*。properties”类型的通配符仅在构建单个模块时起作用,而不是顶级项目(我相信这是一个已知问题)。

问题是如何将非核心模块中定义的属性文件合并到核心模块中定义的现有属性文件中。目前我正在使用BeanPostProcessor - 我只是想知道是否有更简单/更优雅的方法来做到这一点?

由于

3 个答案:

答案 0 :(得分:9)

您可以非常轻松地将多个属性文件收集到一个bean中:

<bean id="allProperties" class="org.springframework.beans.factory.config.PropertiesFactoryBean">
  <property name="singleton" value="true"/>
  <property name="ignoreResourceNotFound" value="true"/>
  <property name="locations">
    <list>
      <value>classpath*:default.properties</value>
      <value>classpath*:overrides.properties</value>
      <value>file:${APP_HOME}/**/*.properties</value>
    </list>
  </property>
</bean>

此特定示例将收集APP_HOME中类路径和属性文件上的所有default.properties,overrides.properties。现在,您可以从ProperyPlaceholderConfigurer或您的自定义后处理器中引用此bean。

答案 1 :(得分:7)

以下代码段应该可以帮助您入门

<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer" id="corePlaceHolder">
    <property name="ignoreUnresolvablePlaceholders" value="true"/>
    <property name="systemPropertiesModeName" value="SYSTEM_PROPERTIES_MODE_OVERRIDE"/>
    <property name="searchSystemEnvironment" value="true"/>
    <property name="locations">
        <list>
            <value>classpath*:config/*/config1/*.properties</value>
            <value>classpath*:config/*/config2/*.properties</value>
            <value>classpath*:config/*/config3/*.properties</value>
            <value>classpath*:custom.properties</value>
        </list>
    </property>
</bean>     

您可以将属性文件存储在以下层次结构中,确保可以通过类路径

访问配置
config
  config1
     a.properties
  config2
     b.properties
  config3
     c.properties
custom.properties

答案 2 :(得分:6)

这就是你需要做的一切:

<context:property-placeholder location="first.properties" order="0" ignore-unresolvable="true"/>
<context:property-placeholder location="second.properties" order="0" ignore-unresolvable="true"/>
<context:property-placeholder location="empty.properties" order="1"/>

问题很简单:如果property-placeholder没有某个属性的值,它将抛出异常,即使存在其他property-placeholder。

该解决方案使用order来了解哪一个property-placeholder并在所有其他ignore-unresolvable="true"上设置property-placeholder,以便每个property-placeholder都有机会提供一个值。在多模块项目中,最后ignore-unresolvable="true"可能为空或提供failafe-defaults。

注意:如果将所有property-placeholder设置为String Spring将只传递您编写的内容而不会抛出异常。当然,如果您希望它不是java.lang.NumberFormatException: For input string: "${something}",那么在格式转换过程中您肯定会得到像order这样的感知。

注意:仅使用具有特定属性值的第一个{property-placeholder order最低的0。如果您要覆盖属性,请使用比1property-override或{{1}}更大的{{1}}范围。

使用Spring 3.2.1测试,但所有提到的属性都存在于3.0中。见JavaDoc of PropertyPlaceholderConfigurer