Spring 表达式语言SpEL

1、SpEL概述

Spring Expression Language(SpEL)是一种强大的表达式语言,支持在运行时查询和操作对象图。它提供了方法调用和基本的字符串模板等功能。

SpEL的创建是为了向spring社区提供单一支持的表达式语言,可以在所有产品中使用SpEL。它的语言特性是由spring项目中的项目需求驱动的,SpEL基于一种与技术无关的API,允许在需要时集成其他表达式语言实现。

SpEL不与spring直接相关,可以独立使用。SpEL表达式语言支持的功能有:文本表达、布尔和关系运算符、正则表达式、类表达式、访问(属性、数组、列表、map)、调用方法、分配、调用构造函数、bean引用、数组构建、内联列表、内联map、三元操作符、变量、用户定义的功能、集合投影、集合选择和模块化表达式。

2、表达式接口

使用SpEL API来评估字符串表达式:

ExpressionParser parser =new SpelExpressionParser();

Expression expression = parser.parseExpression("'hello world'");

String msg = (String) expression.getValue();

消息变量的值是“hello world”。ExpressionParser接口负责解析表达式字符串,表达式字符串是由单引号括起来表示的文本字符串。Expression接口负责评估表达式字符串,当调用parser.parseExpression()和expression.getValue()时,可能会抛出ParseException和EvaluationException异常。

ExpressionParser接口 
Expression接口

\bullet 调用方法:可以在字符串上调用concat方法。

ExpressionParser parser =new SpelExpressionParser();

Expression expression = parser.parseExpression("'hello '.concat('world')");

String msg = (String) expression.getValue();

\bullet 访问属性:可以调用String属性的Bytes。

ExpressionParser parser =new SpelExpressionParser();

Expression expression = parser.parseExpression("'hello'.bytes");

byte[] bytes = (byte[]) expression.getValue();

SpEL还支持使用标点符号的嵌套属性:

ExpressionParser parser =new SpelExpressionParser();

Expression expression = parser.parseExpression("'hello'.bytes.length");

int length = (Integer) expression.getValue();

\bullet 调用构造函数:字符串的构造函数可以被调用,而不是使用字符串。

ExpressionParser parser =new SpelExpressionParser();

Expression expression = parser.parseExpression("new String('hello').toUpperCase()");

String msg = expression.getValue(String.class);

3、对于bean定义的支持

SpEL表达式可以与基于XML或基于注解的配置元数据一起使用来定义BeanDefinitions。这两种方式定义表达式的语法形式是#{<expression string>}。

\bullet 基于XML的配置

使用以下表达式来设置属性或构造器的参数值:

<bean id="testBean" class="com.xxx.TestBean">

    <property name="randomNumber" value="#{T(java.lang.Math).random() * 100}" />

</bean>

使用预定义的变量systemProperties:

<bean id="testBean" class="com.xxx.TestBean">

    <property name="defaultLocale" value="#{systemProperties['user.region']}" />

</bean>

通过名称引用其他bean属性:

<bean id="testBean" class="com.xxx.TestBean">

    <property name="randomNumber" value="#{T(java.lang.Math).random() * 100}" />

</bean>

<bean id="testBean1" class="com.xxx.TestBean1">

    <property name="defaultLocale" value="#{testBean.randomNumber}" />

</bean>

\bullet 基于注解的配置

@Value注解可以放在字段、方法、构造器参数上,用以指定默认值。

等价于以下方式:

自动装配的方法和构造器也可以使用@Value注解:

4、常用表达式

1)文本表达式

文本表达式的类型包括字符串、数值、布尔值和null。字符串由单引号分隔,要将一个单引号本身放在一个字符串中:

2)属性、数组、列表、Map、索引器

属性引用使用句点来指示嵌套的属性值:

属性引用

注意:对于属性名称的第一个字母,不区分大小写。

数组和列表的内容使用中括号表示法来获取:

数组和列表内容获取

Map的内容使用key来获取:

Map的内容使用

3)内联列表

列表可以使用{}表示法直接在表达式中表达:

4)内嵌Map

Map可以使用{key:value}来表达:

5)数组构建

可以使用Java语法来构建数组,或选择提供一个初始化程序在构造时填充该数组:

注意:在构建多维数组时,不允许提供初始化程序。

6)方法

使用Java编程语法调用方法,或调用文字方法,其可变参数也被支持:

7)运算符

关系运算符支持等于、不等于、小于、小于等于、大于、大于等于等操作运算符:

SpEL还支持基于instanceof和正则表达式的匹配运算符:

逻辑运算符:

数学运算符:

8)赋值

属性的设置是通过使用赋值操作符来完成的,可以在setValue调用中完成,也可以在getValue调用中完成。

9)类型

特殊的T运算符可以用来指定java.lang.Class的一个实例,静态方法也使用这个操作符来调用。StandardEvaluationContext使用TypeLocator来查找类型,而StandardTypeLocator是建立在了解java.lang包的基础上的。T()对java.lang中类型的引用不需要完全限定,但所有其他类型引用必须完全限定。

10)构造函数

构造函数可以使用new运算符调用,除基本类型和字符串外,完全限定的类名应该被使用。

11)变量

在表达式中引用变量可以使用语法#variableName,它使用StandardEvaluationContext的setVariable方法设置。

12)函数

可以通过注册在表达式字符串内调用的用户定义函数来扩展SpEL,该函数使用的方法需要在StandardEvaluationContext中注册。

registerFunction定义

13)Bean引用

可以使用@符号从表达式中查找bean。

如果要访问一个工厂bean自身,需要使用&符号。

14)三元运算符

可以使用三元运算符来执行表达式中的if-then-else条件。

15)Elvis运算符

Elvis运算符缩短了三元运算符的语法,在Groovy中使用。使用三元运算符语法,需要重复两次变量:

String s = "aaaa";

String s2 = s != null ? s : "null";

使用Elvis运算符:

16)安全导航运算符

安全导航运算符用于避免NullPointerException,用于Groovy。当有一个对象需要引用时,可能需要在访问对象的方法或属性前验证它是否为空,为了避免这种情况,安全导航操作符将简单的返回null,而不是抛出异常。

17)集合选择

选择允许开发者通过从条目中选择一些源集合转化为另一个。

使用语法.?[selectionExpression],将过滤集合并返回包含原始元素子集的心集合。

18)集合投影

投影允许集合驱动子表达式的评估,结果是一个新的集合。使用语法![projectionExpression]。

19)表达式模板

表达式模板允许将文本与一个或多个评估块混合,每个评估块都用可定义的前缀和后缀字符分隔,如选择是使用#{}作为分隔符。


--参考文献《Srping5开发大全》

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,590评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,808评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,151评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,779评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,773评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,656评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,022评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,678评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,038评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,756评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,411评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,005评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,973评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,053评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,495评论 2 343

推荐阅读更多精彩内容