在Spring XML配置文件中装配Bean的属性和构造参数都是静态的,而在运行期才知道装配的值,就可以使用SpEL实现
SpEL表达式的首要目标是通过计算获得某个值。在计算这个数值的过程中,会使用到其他的值并会对这些值进行操作。
SpEL特性:(1)、使用Bean的ID来引用Bean;(2)、调用方法和访问对象的属性;(3)、对值进行算术、关系和逻辑运算;(4)、正则表达式匹配;(5)、集合操作
#{ }标记会提示 Spring这个标记里的内容是SpEL表达式。
最简单的属性注入:<property name="count" value="#{5}" />
还可以与非SpEL表达式的值混用:<property name="message" value="The value is #{5}" />
浮点数value="#{89.7}" 科学记数法value="#{1e4}"=====>10000.0
String类型的字面值:<property name="name" value="#{'Tom'}" />或<property name="name" value='#{"Tom"}' /> 单引号和双引号相互包含的方式使用
布尔类型:<property name="enable" value="#{false}" />
引用Bean:<property name="userDao" value="#{userDao}" />等价于<property name="userDao" ref="userDao" />
引用Bean属性:<property name="orderOwner" value="#{user.name}" />类似代码:User user=new User();order.setOrderOwner(user.getName());
引用Bean的方法:<property name="currUser" value="#{userDao.getCurrUser()}" />(假设userDao内有公共方法getCurrUser())
现在如果想把userDao.getCurrUser()得到的当前英文用户名转为大写字母:<property name="currUser" value="#{userDao.getCurrUser().toUpperCase()}" />,这时如果得到的用户名为空(null),则会抛NullPointerException异常,为了避免异常,可以使用?.代替点(.),如:<property name="currUser" value="#{userDao.getCurrUser()?.toUpperCase()}" />这样null后面的方法不再执行
SpEL表达式中使用T( )运算符访问指定类的静态方法和常量。
如引用Math类的PI常量:<property name="pi" value="#{T(java.lang.Math).PI}" />
如调用Math类的random静态方法:<property name="randomNumber" value="#{T(java.lang.Math).random()}" />
使用SpEL进行数值运算:
加法:<property name="adjustedAmount" value="#{counter.total+42}" />(把counter Bean的total属性值与42相加,注意+运算符两边必须是数字型)
其他减、乘、整除、求余运算符类似。
使用SpEL进行比较和逻辑表达式运算时,会将得到的布尔型(true或false)赋值给属性,注意,比较运算符的<、<=、>、>=这些符号含有<、>,在XML中有特殊意义,故使用中需要用文本替代方式。==对应eq、<对应lt、<=对应le、>对应gt、>=对应ge
比较运算符:<property name="equal" value="#{counter.total==100}" />
逻辑表达式(and 、or 、not或!):<property name="largeCircle" value="#{shape.kind=='circle' and shape.perimeter gt 10000}" />
条件表达式(使用三元运算符? : ):<property name="song" value="#{kenny.song != null ? kenny.song : 'Greensleeves'}" />
上述条件表达式中kenny.song有重复,可以简化写:<property name="song" value="#{kenny.song ? : 'Greensleeves'}" />
正则表达式:<property name="validEmail" value="#{admin.email matches '/^[a-zA-Z0-9_-]+@[a-zA-Z0-9_-]+(\.[a-zA-Z0-9_-]+)+$/'}" />
在SpEL中筛选集合
假设有个City类,属性有name、state、population
在Spring XML中引入util命名空间,定义一个<util:list>集合
<util:list id="cities">
<bean class="cn.hjp.norClass.City">
<property name="name" value="Chicago"></property>
<property name="state" value="IL"></property>
<property name="population" value="2853114"></property>
</bean>
<bean class="cn.hjp.norClass.City">
<property name="name" value="Atlanta"></property>
<property name="state" value="GA"></property>
<property name="population" value="537958"></property>
</bean>
<bean class="cn.hjp.norClass.City">
<property name="name" value="Dallas"></property>
<property name="state" value="TX"></property>
<property name="population" value="1279910"></property>
</bean>
<bean class="cn.hjp.norClass.City">
<property name="name" value="Houston"></property>
<property name="state" value="TX"></property>
<property name="population" value="2242193"></property>
</bean>
<bean class="cn.hjp.norClass.City">
<property name="name" value="Jal"></property>
<property name="state" value="NM"></property>
<property name="population" value="1996"></property>
</bean>
</util:list>
访问集合成员:<property name="chosenCity" value="#{cities[2]}" /> 选出集合中第三个City
<property name="chosenCity" value="#{cities[T(java.lang.Math).random()*cities.size()]}" />结合SpEL表达式其他方法
[]运算符也可以获取java.util.Map集合中成员,假设City是Map集合:<property name="chosenCity" value="#{cities['Dallas']}" />
从java.util.Properties集合中取值,通过<util:properties>元素加载properties配置文件,假设文件有内容twitter.accessToken=aadkeudnfsje233sdjfs。获取方式:<property name="chosenCity" value="#{settings['twitter.accessToken']}" />
Spring还为SpEL创造了两种特殊的选择属性方式:systemEnvironment和systemProperties
systemEnvironment包含了应用程序所在机器上的所有环境变量(其实是一个java.util.Properties集合),如获取JAVA_HOME路径:<property name="homePath" value="#{systemEnvironment['JAVA_HOME']}" />
systemProperties包含了java应用程序启动时所设置的所有属性(通常通用-D参数)。因此。如果使用-Dapplication.home=/etc/myapp,来启动JVM,那么通过SpEL获取:<property name="homePath" value="#{systemProperties['application.home']}" />
使用SpEL获取字符串某字符:'This is a test'[3]
查询集合成员:
<property name="bigCities" value="#{cities.?[population gt 100000]}" />得到人口大于100000的城市集合
<property name="aBigCities" value="#{cities.^[population gt 100000]}" />得到人口大于100000的第一个符合条件的大城市
<property name="bigCities" value="#{cities.$[population gt 100000]}" />得到人口大于100000的最后一个符合条件的大城市
投影集合:从集合中选取每个成员的属性放入一个新的集合中,使用.![]运算符
<property name="cityNames" value="#{cities.![name]}" /> 得到城市的名字并赋予String类型集合
<property name="cityNames" value="#{cities.![name + ', ' + state]}" />属性拼接,得到如"Chicago,IL"、"Atlanta,GA"、等数据集合
注意:在传统方式难以实现的情况下可以考虑使用SpEL,不要把过多逻辑放入SpEL表达式中