当文件列表是参数时,使用Spring <util:properties>加载多个属性文件时出现问题

时间:2017-03-10 15:39:13

标签: java spring config configuration-files properties-file

我们需要将多个属性文件一起加载并将它们用作属性的一个来源。 <util:properties>允许您传递逗号分隔的文件列表,到目前为止一切正常。所以,以下是好的:

<util:properties loaction="conf/file1.properties,conf/file2.properties,abc-*.properties" />

但是,在我们的示例中,属性文件列表不是固定的,而是来自之前加载的另一个主属性文件。我们希望将该列表作为参数传递给<util:properties>,但它不起作用。

<util:properties location="${allPropertiesFiles}" />

${allPropertiesFiles}定义为

allPropertiesFiles=conf/file1.properties,conf/file2.properties,abc-*.properties

由于文件列表中有逗号,因此失败。它将它们视为单个文件名并抛出FileNotFoundException。

我想知道Spring在什么时候试图用逗号分割文件,看起来它在解析$ {allPropertiesFiles}之前就已经发生了。例如,如果我这样做,它工作正常,但这对我们来说不是一个实际的解决方案,因为我们不知道该列表中包含了多少文件。

<util:properties location="${propFile.location1},${propFile.location2},${propFile.location3}" />

更新:

这似乎是一个Spring问题,处理和拆分&#39;,&#39;在解析${...}中的属性值之前。我甚至尝试使用Spring EL来分割它,但它在解析有效的EL时再次失败,因为它首先基于&#39;,&#39;然后评估表达式。下面的示例因EL解析异常而失败:

<util:properties location="#{'${allPropertiesFiles}'.split(',')}" />

仅供参考,这个观察是在Spring 4.2.x中进行的。任何建议都非常感谢。

2 个答案:

答案 0 :(得分:0)

您是否尝试用双引号包围变量值? 在一个小小的测试中,这就像一个魅力:

的ApplicationContext:

<bean id="test" class="de.de.proptest.Test">
    <property name="p" ref="props" />
</bean>
<util:properties location="${props}" id="props" />

我有2个属性文件,a.properties包含内容:

a=1

和b.properties with content:

b=2

使用JVM参数

-Dprops="file:/home/dominik/sandbox/proptest/a.properties, file:/home/dominik/sandbox/proptest/b.properties"

我得到以下输出(切入有趣的点):

Mar 11, 2017 1:32:11 PM org.springframework.beans.factory.config.PropertiesFactoryBean loadProperties
INFO: Loading properties file from URL [file:/home/dominik/sandbox/proptest/a.properties]
Mar 11, 2017 1:32:11 PM org.springframework.beans.factory.config.PropertiesFactoryBean loadProperties
INFO: Loading properties file from URL [file:/home/dominik/sandbox/proptest/b.properties]
Test [p={b=2, a=1}]

答案 1 :(得分:0)

所以,我找到了我的问题的解决方案/解决方法,并希望在此处分享。

util:propertiescontext:property-placeholder的bean解析器将location属性的给定字符串值拆分为,。它在财产解决发生之前发生。因此,如果要将以逗号分隔的属性文件列表传递给这些bean,它将无法工作。

因此,我决定直接使用<util:properties>并将位置设置为参数,而不是使用PropertiesFactoryBean。这修复了如何在spring中构建bean定义,因为它使用Spring默认的bean解析器。然后我提供了一个自定义版本的Spring ResourceArrayPropertyEditor,它将String转换为Resource[]。属性编辑器在转换为Resource[]时处理以逗号分隔的字符串。

以下是我的上下文xml:

    <bean class="org.springframework.beans.factory.config.CustomEditorConfigurer">
        <property name="customEditors">
            <map>
                <entry key="org.springframework.core.io.Resource[]" value="com.mycompany.CustomResourceArrayPropertyEditor"/>
            </map>
        </property>
    </bean> 
    <bean id="allPropertiesFromFiles" class="org.springframework.beans.factory.config.PropertiesFactoryBean">
            <property name="locations" value="${propertiesFile.locations}" />
    </bean>

这是我的自定义PropertyEditor

public class CustomResourceArrayPropertyEditor extends ResourceArrayPropertyEditor
{
  private final ResourcePatternResolver resourcePatternResolver;

  public CustomResourceArrayPropertyEditor()
  {
    this(new PathMatchingResourcePatternResolver());
  }

  public CustomResourceArrayPropertyEditor(ResourcePatternResolver resourcePatternResolver)
  {
    super(resourcePatternResolver, null);
    this.resourcePatternResolver = resourcePatternResolver; 
  }

  @Override
  public void setAsText(String text)
  {
    String pattern = resolvePath(text).trim();
    String[] resourceNames = StringUtils.commaDelimitedListToStringArray(pattern);
    List<Resource> resourceList = new ArrayList<Resource>();
    try {
      for (String resourceName: resourceNames)
      {
        Resource[] resources = resourcePatternResolver.getResources(resourceName);
        for (Resource res: resources)
          resourceList.add(res);
      }
    }
    catch (IOException ex) {
      throw new IllegalArgumentException("Could not resolve resource location pattern [" + pattern + "]", ex);
    }

    setValue(resourceList.toArray(new Resource[0]));
  }
}

我能想到的另一个解决方法是创建一个BeanFactoryPostProcessor来访问bean并更新util:propertiescontext:propery-placeholder的bean定义,以便TypedStringValue用于{{ 1}}属性而不是locations

希望它有所帮助。