Я использую Spring ApplicationContextInitializer для добавления PropertySources в среду. Я проверил, что эта часть работает так же, как и при загрузке свойств в PropertySources. Часть, с которой я борюсь, - это выяснить, как получить доступ к значениям свойств, которые были загружены в ApplicationContextInitializer в конфигурации Spring XML, а также в самом Java-коде.
Вот фрагмент моего ApplicationContextInitializer:
public class ExternalPropertiesApplicationContextInitializer implements ApplicationContextInitializer<ConfigurableApplicationContext> {
@Override
public void initialize(ConfigurableApplicationContext applicationContext) {
Properties properties = load_properties_from_external_source();
PropertiesPropertySource propertySource = new PropertiesPropertySource("external", properties);
applicationContext.getEnvironment().getPropertySources().addFirst(propertySource);
}
}
Таким образом, это загружает мои свойства, у которых есть пары ключей, значений:
jdbc.driverClassName=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/mydb
jdbc.username=user
jdbc.password=passwd
Мой конфигурационный файл Spring выглядит следующим образом:
<context:component-scan base-package="com.example"/>
<context:property-placeholder ignore-resource-not-found="true" system-properties-mode="OVERRIDE"/>
<bean id="myds" class="org.apache.tomcat.jdbc.pool.DataSource" destroy-method="close">
<property name="driverClassName" value="${jdbc.driverClassName}"/>
<property name="url" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
</bean>
Это не работает, потому что он не распознает ${jdbc.driverClassName}
. Я также попробовал #{external.jdbc.driverClassName}
который тоже не работает. Первое не удается, потому что я не загружаю jdbc.properties в свой конфигурационный файл Spring (это намерение). Последний не работает, потому что он не может найти external
bean-компонент. Какой шаг я пропущу? Как вывести значения свойств, которые были загружены извне?
У меня также есть класс Java:
package com.example;
@Component
public class MyClass {
@Value("${jdbc.url}")
private String jdbcUrl;
}
Как и в конфигурации Spring, как мне получить доступ к значениям свойств jdbc.url, которые были загружены при инициализации?
О, и вот фрагмент файла web.xml:
<context-param>
<param-name>contextInitializerClasses</param-name>
<param-value>com.example.ExternalPropertiesApplicationContextInitializer</param-value>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
Было несколько проблем с конфигурационным файлом Spring, который был виновником.
В частности, в заголовке у меня было:
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd">
несмотря на то, что он весной 3.1. Это привело к тому, что Spring вернулась к старой PropertyPlaceholderConfigurer, а не к новому PropertySourcesPlaceholderConfigurer.
Как только я изменил ссылку 3.0.xsd на 3.1.xsd, он начал решать проблему. Новый заголовок выглядит так:
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.1.xsd">
Второй вопрос:
<context:property-placeholder ignore-resource-not-found="true" system-properties-mode="OVERRIDE"/>
Из-за атрибута system-properties-mode Spring снова вернулась к старой PropertyPlaceholderConfigurer, а не к новому PropertySourcesPlaceholderConfigurer.
При использовании старого PropertyPlaceholderConfigurer он не извлекает значения из новой среды, поэтому значения свойств не собирались в конфигурации Spring Context.