背景
类似于datasource.properties之类的配置文件,最初通过Java的Properties类进行处理。这种方式有许多弊端,如每次都需要读取配置文件;若将Properties作为成员变量,则当配置文件缺失时,可能直接会导致程序运行失败。
使用Spring读取并装配配置文件,则可以避免上述麻烦。思路是将配置文件装配到一个具体类上,使用配置项的时候可通过该类的get()方法即可获取。
Java类
如配置类BspAuthConfig.java类如下:
package com.inspur.analysis.tool.user.config;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
/**
* Created by liutingna on 2017/8/22.
*/
@Component("bspAuthConfig")
public class BspAuthConfig {
@Value("#{bspAuthConfigBean['analysis-tool-authentification']}")
private String bspAuthUrl;
public String getBspAuthUrl() {
return bspAuthUrl;
}
public void setBspAuthUrl(String bspAuthUrl) {
this.bspAuthUrl = bspAuthUrl;
}
}
以上,@Component注解即表示装配生成的配置类BspAuthConfig的名字bspAuthConfig;
@Value注解,对应.properties配置文件中的配置项,如取属性配置Bean名称为bspAuthConfigBean的analysis-tool-authentification配置项;
属性配置bspAuthConfigBean通过Spring配置类进行配置,如下所示。
Spring配置
Spring配置有两个作用,一是读取.properties配置文件,并装配为名为bspAuthConfigBean的Bean;二是扫描自定义的配置类BspAuthConfig。
<!--BSP认证配置-->
<context:component-scan base-package="com.inspur.analysis.tool.user.config"/>
<bean id="bspAuthConfigBean" class="org.springframework.beans.factory.config.PropertiesFactoryBean">
<property name="locations">
<array>
<value>classpath:conf.properties</value>
</array>
</property>
</bean>
对应的一个.properties配置文件示例如下:
analysis-tool-authentification=http://localhost:8081/analysis-tool-authentification
应用
子需要配置类的地方,直接创建成员变量,然后使用get方法获取配置项。如:
private static BspAuthConfig bspAuthConfig=null;
static {
bspAuthConfig= ContextUtils.getBean(BspAuthConfig.class, "bspAuthConfig");
}
这里使用了一个工具类ContextUtils,该类比较简单,即获取上下文中的Bean对象,如下:
package com.inspur.analysis.tool.common.utils;
import org.apache.commons.beanutils.BeanUtils;
import org.apache.commons.lang3.StringUtils;
import org.loushang.framework.util.SpringContextHolder;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* 上下文协助
* @author xie_yh
*
*/
public class ContextUtils {
private static ApplicationContext defaultContext = null;
/**
* 获取context
* @return
*/
public static ApplicationContext getContext(){
if(defaultContext != null){
return defaultContext;
}
return SpringContextHolder.getApplicationContext();
}
/**
* 获取bean
* @param name
* @return
*/
public static Object getBean(String name){
return getContext().getBean(name);
}
/**
* 获取bean
* @param clz
* @return
*/
public static <T> T getBean(Class<T> clz){
return getContext().getBean(clz);
}
/**
* 获取bean
* @param clz
* @param name
* @return
*/
public static <T> T getBean(Class<T> clz,String name){
return getContext().getBean(name, clz);
}
/**
* 获取值
* @param name
* @param keyname
* @param defaultValue
* @return
*/
public static String getValue(String name,String keyname,String defaultValue){
Object bean = getBean(name);
try {
return StringUtils.defaultIfBlank(BeanUtils.getProperty(bean, keyname),defaultValue);
} catch (Exception e) {
}
return defaultValue;
}
/**
* 初始化脚本
* @param xmls
*/
public void initContext(String[] xmls){
//加载spirng配置文件
defaultContext= new ClassPathXmlApplicationContext(xmls);
}
}
注意
需要注意的是配置文件key的写法,上述示例中使用中括号:
@Value("#{bspAuthConfigBean['analysis-tool-authentification']}")
另外还可以使用点号 . 的方式,如:
@Value("#{cmspconfig.cmsp_ip}")
但是这是要求配置文件key不能以点号 . 或者中划线 - 等字符分割。
改进
上述配置,当properties配置文件缺少相应的配置项时,会造成注解失败。而按照需求,当缺少properties配置文件时,使用默认值替代,那么可以使用以下方式进行配置。
(注意:这里展示另一个例子,相关Java类与配置文件与上面并不一致)
- Java配置类
配置类@Value注解中,使用美元符号($)读取默认项,且直接使用properties文件中的配置项即可,无需再使用xml中配置的bean的id。(即无需propertyConfigurer.file_storage_server_type或propertyConfigurer['file_storage_server_type'],直接使用file_storage_server_type即可。)
当读取不到配置项时,使用org.springframework.beans.factory.config.PreferencesPlaceholderConfigurer填充默认值,只需要在配置项后面加上冒号和默认值即可。如下默认值为空:
package com.inspur.analysis.tool.dispatcher.ontology.document.config;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
@Component("fileStorageConfig")
public class FileStorageConfig {
@Value("${file_storage_server_type:}")
private String fileStorageServerType;
@Value("${file_storage_server_ip:}")
private String fileStorageServerIp;
@Value("${file_storage_server_port:}")
private String fileStorageServerPort;
@Value("${file_storage_server_username:}")
private String fileStorageServerUsername;
@Value("${file_storage_server_password:}")
private String fileStorageServerPassword;
@Value("${file_storage_server_path:}")
private String fileStorageServerPath;
public String getFileStorageServerType() {
return fileStorageServerType;
}
public void setFileStorageServerType(String fileStorageServerType) {
this.fileStorageServerType = fileStorageServerType;
}
public String getFileStorageServerIp() {
return fileStorageServerIp;
}
public void setFileStorageServerIp(String fileStorageServerIp) {
this.fileStorageServerIp = fileStorageServerIp;
}
public String getFileStorageServerPort() {
return fileStorageServerPort;
}
public void setFileStorageServerPort(String fileStorageServerPort) {
this.fileStorageServerPort = fileStorageServerPort;
}
public String getFileStorageServerUsername() {
return fileStorageServerUsername;
}
public void setFileStorageServerUsername(String fileStorageServerUsername) {
this.fileStorageServerUsername = fileStorageServerUsername;
}
public String getFileStorageServerPassword() {
return fileStorageServerPassword;
}
public void setFileStorageServerPassword(String fileStorageServerPassword) {
this.fileStorageServerPassword = fileStorageServerPassword;
}
public String getFileStorageServerPath() {
return fileStorageServerPath;
}
public void setFileStorageServerPath(String fileStorageServerPath) {
this.fileStorageServerPath = fileStorageServerPath;
}
}
- xml配置文件
该xml需要在Web项目启动时,在上下文参数中加载。
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
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:component-scan base-package="com.inspur.analysis.tool.dispatcher.**.config"/>
<bean id="fileStorageConfigBean" class="org.springframework.beans.factory.config.PropertiesFactoryBean">
<property name="locations">
<array>
<value>classpath:datasource.properties</value>
</array>
</property>
</bean>
<bean id="propertyConfigurer" class="org.springframework.beans.factory.config.PreferencesPlaceholderConfigurer">
<property name="properties" ref="fileStorageConfigBean"/>
</bean>
</beans>
改进2
参见参考资料4.
- Java配置
默认值支持字符串、整型等多种类型:
@Value("#{fileDatasourceConfig['file_ds_server_type']?:1}")
private int serverType;
@Value("#{fileDatasourceConfig['file_ds_server_ip']?:''}")
private String serverIP;
@Value("#{fileDatasourceConfig['file_ds_server_port']?:21}")
private int serverPort;
@Value("#{fileDatasourceConfig['file_ds_server_username']?:''}")
private String username;
@Value("#{fileDatasourceConfig['file_ds_server_password']?:''}")
private String password;
@Value("#{fileDatasourceConfig['file_ds_server_path']?:''}")
private String structuredFilePath;
- xml配置
使用util:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:beans="http://www.springframework.org/schema/beans"
xmlns:p="http://www.springframework.org/schema/p"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xmlns:task="http://www.springframework.org/schema/task"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:util="http://www.springframework.org/schema/util"
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
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc-3.1.xsd
http://www.springframework.org/schema/task
http://www.springframework.org/schema/task/spring-task-3.1.xsd http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util.xsd">
<!--文档数据源配置类-->
<context:component-scan base-package="com.inspur.analysis.tool.datasource.**.config"/>
<bean id="fileDatasourceConfig" class="org.springframework.beans.factory.config.PropertiesFactoryBean">
<property name="locations">
<array>
<value>classpath:datasource.properties</value>
</array>
</property>
</bean>
<bean id="fileDsRefConf" class="org.springframework.beans.factory.config.PreferencesPlaceholderConfigurer">
<property name="properties" ref="fileDatasourceConfig"/>
<property name="ignoreUnresolvablePlaceholders" value="true"/>
</bean>
<!--以下加载方式对项目中datasource.xml配置加载数据源datasource.properties文件有些影响-->
<!--<util:properties id="fileDatasourceConfig" location="classpath:datasource.properties"/>-->
</beans>
参考资料
【1】spring中@value注解需要注意
【2】给Spring的placeholder设置默认值
【3】spring报“Could not resolve placeholder”错误
【4】Spring @Value default value