Spring(13)——PropertyPlaceholderConfigurer
13 PropertyPlaceholderConfigurer
PropertyPlaceholderConfigurer是对BeanFactoryPostProcessor接口的一种实现,其允许我们在定义bean的时候将bean的某些属性定义为变量形式,然后由PropertyPlaceholderConfigurer在这些bean被实例化之前利用外部定义的属性文件或者其内部定义的属性等将这些变量替换成对应的属性值。
我们可能经常会见到或者用到这样的定义,context:property-placeholder是Spring context命名空间内置的一个标签,针对该标签Spring内部会建立一个PropertyPlaceholderConfigurer对应的bean,其中location属性表示需要加载的外部属性文件对应的路径。
<context:property-placeholder location="/WEB-INF/config/jdbc.properties" /><bean id="dataSource" class="org.apache.commons.dbcp2.BasicDataSource"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>
我们可以看到在上述配置中,id为dataSource的bean对应的属性值我们都是用的变量表示的。我们定义的PropertyPlaceholderConfigurer将在所有bean初始化前利用我们定义的外部属性文件jdbc.properties来对这些变量进行替换,所以对应的值我们可以直接定义在jdbc.properties文件中。
jdbc.driverClassName=oracle.jdbc.driver.OracleDriver jdbc.url=jdbc:oracle:thin:@localhost:1521:orcl jdbc.username=crm jdbc.password=crm
我们可以在bean容器中直接定义一个PropertyPlaceholderConfigurer类型的bean,用以在所有的bean定义加载完成后利用其可用的属性替换bean定义中使用的变量。PropertyPlaceholderConfigurer和PropertySourcesPlaceholderConfigurer均是继承自抽象类PlaceholderConfigurerSurpport的,它们使用的变量替换的主逻辑均来自PlaceholderConfigurerSurpport。
13.1 可用来作为替换变量的属性
13.1.1 外部属性文件
PropertyPlaceholderConfigurer常用来替换变量的属性通常是来自外部属性文件定义,通过其setLocation()方法或者是setLocations()方法进行注入,其中setLocation()方法用来注入一个外部属性文件,而setLocations()则是用来注入一个或多个外部属性文件,在注入的时候它们都将会被作为一个Resource进行注入。
通过setLocation()注入单个属性文件。
<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"> <!-- 通过setLocation()方法指定单个外部属性文件的位置 --><property name="location" value="classpath:t1.properties"/></bean>
通过setLocations()注入一个或多个属性文件。
<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"><!-- 通过setLocations()方法指定一到多个外部属性文件的位置 --><property name="locations"><array><value>classpath:t1.properties</value><value>classpath:t2.properties</value></array></property></bean>
13.1.2 内部属性
内部属性是通过setProperties()或setPropertiesArray()方法进行定义的,其中前者接收一个Properties对象,而后者则是接收一个Properties数组。
<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"><!-- 直接通过setProperties()方法定义属性 --><property name="properties"><props><prop key="p1">v1</prop><prop key="p2">v2</prop></props></property></bean>
13.1.3环境变量、系统变量
环境变量、系统变量是可以直接被PropertyPlaceholderConfigurer用来作为属性源对bean定义中使用的变量进行替换的。如JAVA_HOME、user.dir等。环境变量即通过System.getenv(name)可获取到的,而系统变量则表示指定的JVM参数,即通过System.getProperty(name)可获取到的。
13.1.4 属性来源优先级
既然我们可以用来替换变量的属性来源有多种不同的方式,那么当其中两者存在冲突时将采取怎样的策略呢?
外部属性文件定义和内部属性定义算是一类,这里我们把它定义为A类,环境变量和系统变量定义为B类。在A类里面当外部属性文件中的定义与内部属性的定义存在冲突,即存在相同的属性时,默认情况下是外部属性文件定义的属性值将覆盖内部properties定义的。如果需要设置内部属性定义将覆盖外部属性文件定义的属性,则可以通过setLocalOverride()方法指定对应的localOverride为true来达到设置内部属性定义覆盖外部属性文件定义的属性的目的。
<!-- 通过setLocalOverride()方法设置内部属性定义将覆盖外部属性文件的定义 --><beanclass="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"p:localOverride="true"></bean>
对于B类存在与A类相同的属性定义时,将采取何种策略也是根据一个参数来确定的。通过setSystemPropertiesMode()方法可以指定对应的值,也可以通过setSystemPropertiesModeName()来指定对应的策略名称。其中可选值和可选名称如下:
- SYSTEM_PROPERTIES_MODE_NEVER:对应值为0。表示不使用系统属性进行替换。
- SYSTEM_PROPERTIES_MODE_FALLBACK:对应的值为1,这是默认选项。表示只有在A类属性中没有找到变量对应的属性时才会尝试使用系统属性来进行替换。
- SYSTEM_PROPERTIES_MODE_OVERRIDE:对应的值为2。表示当系统属性存在变量对应的属性时将使用系统属性的值。
<!-- 指定系统属性将具有最高优先级 --><beanclass="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"p:systemPropertiesMode="2"><property name="properties"><props><prop key="java.tempdir">abc</prop></props></property></bean>
13.2 变量的定义
变量默认是使用“${varName}”的形式进行定义的,即以“${”开始,以“}”结束,中间对应变量名,PropertyPlaceholderConfigurer默认会将这种形式的定义当做是需要进行替换的变量定义,中间对应的是变量名。基本上除了id和name以外,bean定义的其它内容都可以使用变量进行定义,如class、parent等。
<!-- 将userDir定义为一个需要替换的变量user.dir --><bean id="hello" class="com.app.Hello" p:userDir="${user.dir}"/>
以上就是一个将属性userDir定义为变量user.dir的示例,在bean该bean被实例化之前将会由定义的PropertyPlaceholderConfigurer将其中的变量${user.dir}进行替换。如果我们不想变量定义为“${varName}”的形式我们也可以通过setPlaceholderPrefix()方法来指定变量对应的前缀,默认是“${”,然后通过setPlaceholderSuffix()方法来指定对应的后缀,默认是“}”。所以如果我们希望我们的变量定义为“@{varName}”的形式,我们可以如下定义:
<!-- 指定变量的前缀为“@{”,后缀为“}” --><beanclass="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"p:placeholderPrefix="@{" p:placeholderSuffix="}" />
那么对应的变量就可以定义成如下这样:
<!-- 将userDir定义为一个需要替换的变量user.dir --><bean id="hello" class="com.app.Hello" p:userDir="@{user.dir}" /> </beans>
13.3 指定默认值
我们还可以在定义属性变量时指定对应的默认值。如果指定了属性变量的默认值,则在未找到可用于替换当前属性变量的属性时将使用定义好的默认值来替换当前属性变量。我们在定义属性变量时在变量名后面紧接着就可以指定默认值,默认值与变量名之间以分号隔开,形式如:${varName:defValue},其中varName表示变量名,defValue表示默认值。
<!-- 指定属性变量maxVal的默认值为100 --><bean id="hello" class="com.app.Hello" p:maxVal="${maxVal:100}" />
属性变量名与默认值之间的分隔符默认是分号“:”,如果有需要用户也可以通过PropertyPlaceholderConfigurer的setValueSeparator()方法进行修改。
<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"> <!-- 指定属性变量名与默认值之间的分隔符为两个分号 --><property name="valueSeparator" value="::"/></bean>
13.4 忽略文件未找到
默认情况下我们在通过setLocation()或setLocations()方法指定外部属性文件时,如果对应的文件不存在将抛出异常信息。通过setIgnoreResourceNotFound()方法我们可以设置是否忽略文件未找到的情况,默认为false,即抛出异常信息。如果用户希望忽略对应的错误,则可以设置对应的值为null。
<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"> <property name="location" value="classpath:t12.properties"/> <!-- 指定当外部属性文件不存在时不抛出异常 --><property name="ignoreResourceNotFound" value="true"/></bean>
13.5 忽略变量不能解析
默认情况下PropertyPlaceholderConfigurer会把bean定义中所有${varName}形式的定义都当做是一个属性变量定义进行替换。当bean定义中存在形式为${varName}的变量定义但是又不能被PropertyPlaceholderConfigurer利用定义的外部属性文件包含的属性或内部属性定义的属性或环境变量进行替换时,其会抛出异常信息。如果希望在对应的属性变量不能被解析时不抛出异常信息,则可以通过setIgnoreUnresolvablePlaceholders()方法指定ignoreUnresolvablePlaceholders的值为true,这样就将忽略变量不能被解析的情况。
<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"> <property name="location" value="classpath:t12.properties"/> <!-- 指定当外部属性文件不存在时不抛出异常 --><property name="ignoreResourceNotFound" value="true"/><!-- 忽略变量不能被解析的情况 --><property name="ignoreUnresolvablePlaceholders" value="true"/></bean>
Spring(13)——PropertyPlaceholderConfigurer相关推荐
- Spring(五):SpringStruts2Hibernate整合后,实现查询Employee信息
背景: 基于之前两篇文章<Spring(三):Spring整合Hibernate>.<Spring(四):Spring整合Hibernate,之后整合Struts2>,了解了如 ...
- 『高级篇』docker之开发用户服务EdgeService(13)
原创文章,欢迎转载.转载请注明:转载自IT人故事会,谢谢! 原文链接地址:『高级篇』docker之开发用户服务EdgeService(13) 上一节开发了用户服务,即将开发的是用户服务EdgeServ ...
- Java进击框架:Spring(一)
Java进击框架:Spring(一) 前言 创建Spring项目 Spring IoC容器和Beans介绍 Bean的概述 Spring IoC 配置元数据 实例化Bean 依赖注入 循环依赖 详细配 ...
- Spring(二)IOC容器的初始化流程
文章目录 一.Spring 核心容器类 1.1 BeanFactory 1.2 ApplicationContext 1.3 BeanDefinition 二.IOC容器的初始化 2.1 基于Xml的 ...
- 乐优商城(13)--购物车
乐优商城(13)–购物车 一.搭建购物车服务 1.1.创建module 1.2.pom依赖 <?xml version="1.0" encoding="UTF-8& ...
- Spring(07)——单例注入多例之lookup-method
2019独角兽企业重金招聘Python工程师标准>>> Spring(07)--单例注入多例之lookup-method 博客分类: spring 7 单例注入多例之lookup-m ...
- FPGA逻辑设计回顾(13)RAM以及ROM的IP核定制以及关键参数
文章目录 前言 RAM IP的定制 Xilinx的IP定制位置 Block RAM的定制过程 第一页 第二页 第三页 第四页 第五页 Block RAM的延迟讨论 ROM IP核的定制 总结 前言 本 ...
- 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(13)-系统日志和异常的处理③
构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(13)-系统日志和异常的处理③ 参考文章: (1)构建ASP.NET MVC4+EF5+EasyUI+Unity ...
- Xamarin图表开发基础教程(13)OxyPlot框架支持的其它图表
Xamarin图表开发基础教程(13)OxyPlot框架支持的其它图表 除了以上提到的图表外,OxyPlot组件还包含了6种类型的其它图表,分别为等高线图.箱线图.饼图.热图.散点图和散点误差图,如图 ...
- python 简单图像处理(13) 二值图腐蚀和膨胀,开运算、闭运算
原文:http://www.cnblogs.com/xianglan/archive/2010/12/29/1921211.html python 简单图像处理(13) 二值图腐蚀和膨胀,开运算.闭运 ...
最新文章
- boost::mpi模块is_mpi_op 功能的测试
- 【Linux】一步一步学Linux——ifcfg命令(157)
- BZOJ.4553.[HEOI2016TJOI2016]序列(DP 树状数组套线段树/二维线段树(MLE) 动态开点)
- 清华博导尹首一, 带你吃透 AI 芯片来龙去脉!
- forward内部跳转 和redirect重定向跳转的区别
- ROS的学习(十八)使用rosserial创建一个publisher
- 3. 什么是icmp?icmp与ip的关系_0.3亿人口的美国会比3亿人口的美国富裕吗?
- Ubuntu18.04TLS运行linux版百度网盘客户端的问题
- java lpad oracle_Oracle中Lpad函数和Rpad函数的用法
- provision文件路径
- 高分一号/二号/六号定标系数
- 计算机退回登录界面,win7开机怎么自动登录用户?Win进入桌面又返回登录界面故障解决...
- 20162312Java结对编程之挑战出题
- 虹科ELPRO - EMS系统实现苏州某医药仓库温湿度自动监测 - 100% GxP合规
- java学习笔记 java编程思想 第6章 访问权限控制
- [黑马IOS自学第九篇]OC语言基础,面向对象概念,类的创建,OC多文件开发等
- 分段二次插值的matlab程序,一维优化方法之二次插值法matlab程序
- 怎么将微信小程序项目转为uniapp项目
- fastdfs工作原理
- 推荐16个免费开源的商业智能工具