SpEL表达式

SpEL概述

Spring表达式语言全称为"Spring Expression Language",缩写为"SpEL",类似于Struts2x中使用的OGNL表达式语言,能在运行时构建复杂表达式、存取对象图属性、对象方法调用等等,并且能与Spring功能完美整合,如能用来配置Bean定义。

表达式语言给静态Java语言增加了动态功能。

SpEL是单独模块,只依赖于core模块,不依赖于其他模块,可以单独使用。

SpEL能干什么?

表达式语言一般是用最简单的形式完成最主要的工作,减少我们的工作量。

SpEL支持如下表达式:

一、基本表达式

字面量表达式、关系,逻辑与算数运算表达式、字符串连接及截取表达式、三目运算及Elivis表达式、正则表达式、括号优先级表达式

二、类相关表达式

类类型表达式、类实例化、instanceof表达式、变量定义及引用、赋值表达式、自定义函数、对象属性存取及安全导航表达式、对象方法调用、Bean引用

三、集合相关表达式

  • 内联List、内联数组、集合,字典访问、列表,字典,数组修改、集合投影、集合选择
  • 不支持多维内联数组初始化
  • 不支持内联字典定义

四、其他表达式

模板表达式。

注:SpEL表达式中的关键字是不区分大小写的。

SpEL语法

一、基本表达式

1.字面量表达式

SpEL支持:字符串数字类型(int、long、float、double)布尔类型null类型

2.算数运算表达式

SpEL支持:加(+)减(-)乘(*)除(/或div)求余(%或mod)幂(^)运算。

3.关系表达式

SpEL支持:等于(==或eq)不等于(!=或ne)大于(>或gt)大于等于(>=或ge)小于(<或lt)小于等于(<=或le)区间(between)运算。

4.逻辑表达式

SpEL支持:且(and或&&)或(or或||)非(!或NOT)

5.字符串连接及截取表达式

使用"+"进行字符串连接,如"'Hello ' + 'World!'"->"Hello World!"

使用"'String'[index]"来截取一个字符,目前只支持截取一个,如"'Hello World!'[0]"->"H"

6.三目运算

三目运算符“表达式1?表达式2:表达式3”用于构造三目运算表达式,如“2>1?true:false”->true

7.Elivis运算符

Elivis运算符“表达式1?:表达式2”Groovy语言引入用于简化三目运算符的,当表达式1为非null时则返回表达式1,当表达式1为null时则返回表达式2,简化了三目运算符方式“表达式1? 表达式1:表达式2”,如“null?:false”->false,而“true?:false”->true

如:name != null? name : "other"简写为:name?:"other"

8.正则表达式

使用“str matches regex,如“'123' matches '\d{3}'”->true

9.括号优先级表达式

使用“(表达式)”构造,括号里的具有高优先级

二、类相关表达式

1.类类型表达式

使用“T(Type)”来表示java.lang.Class实例,Type必须是类全限定名java.lang包除外,即该包下的类可以不指定包名;使用类类型表达式还可以进行访问类静态方法及类静态字段。可以进行静态字段访问如“T(Integer).MAX_VALUE”;也可以进行静态方法访问如“T(Integer).parseInt('1')”

2.类实例化

类实例化同样使用java关键字“new”,类名必须是全限定名,但java.lang包内的类型除外,如StringInteger。如new String('lxy')new java.util.Date()

3.instanceof表达式

SpEL支持instanceof运算符,跟Java内使用同义;如'haha' instanceof T(String)->true

4.变量定义及引用

变量定义通过EvaluationContext接口的setVariable(variableName, value)方法定义;在表达式中使用"#variableName"引用;

除了引用自定义变量,SpE还允许引用根对象及当前上下文对象,使用"#root"引用根对象,使用"#this"引用当前上下文对象

5.自定义函数

目前只支持类静态方法注册为自定义函数

6.表达式赋值

使用Expression#setValue方法可以给表达式赋值

7.对象属性存取及安全导航表达式

对象属性获取非常简单,即使用如“a.property.property”这种点缀式获取,SpEL对于属性名首字母不区分大小写的;

SpEL还引入了Groovy语言中的安全导航运算符“(对象|属性)?.属性”,用来避免“?.”前边的表达式为null时抛出空指针异常,而是返回null;修改对象属性值则可以通过赋值表达式或Expression接口的setValue方法修改。

8.对象方法调用

对象方法调用更简单,跟Java语法一样;如'haha'.substring(2,4)->ha;而对于根对象可以直接调用方法;

9.Bean引用

SpEL支持使用“@”符号来引用Bean,在引用Bean时需要使用BeanResolver接口实现来查找Bean,Spring提供BeanFactoryResolver实现。

三、集合相关表达式

1.内联List

从Spring3.0.4开始支持内联List,使用{表达式,...}定义内联List,如“{1,2,3}”将返回一个整型的ArrayList,而“{}”将返回空的List,对于字面量表达式列表,SpEL会使用java.util.Collections.unmodifiableList方法将列表设置为不可修改

2.内联数组

和Java 数组定义类似,只是在定义时进行多维数组初始化。

3.集合,字典元素访问

SpEL目前支持所有集合类型字典类型的元素访问,使用"集合[索引]"访问集合元素,使用"map['key']"访问字典元素

4.列表,字典,数组元素修改

可以使用赋值表达式或Expression接口的setValue方法修改

5.集合投影

在SQL中投影指从表中选择出列,而在SpEL指根据集合中的元素中通过选择来构造另一个集合,该集合和原集合具有相同数量的元素;SpEL使用"(list|map).![投影表达式]"来进行投影运算:如"#list.![#this+1]"即每个元素+1、"#map.![value+1]"即map的每个value+1。

对于集合或数组使用如上表达式进行投影运算,其中投影表达式中“#this”代表每个集合或数组元素,可以使用比如“#this.property”来获取集合元素的属性,其中“#this”可以省略。

Map投影最终只能得到List结果,如上所示,对于投影表达式中的"#this"将是Map.Entry,所以可以使用"value"来获取值,使用"key"来获取键。

6.集合选择

在SQL中指使用select进行选择行数据,而在SpEL指根据原集合通过条件表达式选择出满足条件的元素并构造为新的集合,SpEL使用"(list|map).?[选择表达式]",其中选择表达式结果必须是boolean类型,如果true则选择的元素将添加到新集合中,false将不添加到新集合中。如#list.?[#this>4],使用“#this”表示当前元素、"#map.?[key!='a']""#map.?[key!='a'].![value+1]"

对于字典选择,如“#map.?[#this.key != 'a']”将选择键值不等于”a”的,其中选择表达式中“#this”Map.Entry类型,而最终结果还是Map,这点和投影不同;集合选择和投影可以一起使用,如“#map.?[key != 'a'].![value+1]”将首先选择键值不等于”a”的,然后在选出的Map中再进行“value+1”的投影。

7.表达式模板

模板表达式就是由字面量与一个或多个表达式块组成。每个表达式块由“前缀+表达式+后缀”形式组成,如“${1+2}”表达式块。如“Error ${#v0} ${#v1}”表达式表示由字面量“Error ”、模板表达式“#v0”、模板表达式“#v1”组成,其中v0和v1表示自定义变量,需要在上下文定义。

解析表达式的时候需要指定模板,模板通过ParserContext接口来定义

在Bean定义中使用spel表达式

注解风格的配置(@Value)

基于注解风格的SpEL配置也非常简单,使用@Value注解来指定SpEL表达式,该注解可以放到字段方法方法参数上。

测试Bean类如下,使用@Value来指定SpEL表达式

1
2
3
4
public class SpELBean {  
@Value("#{'Hello World'}")
private String value;
}

更多SpEL介绍及案例