Spring表达式基础(Spring Expression Language)

使用Spring表达式语言进行装配

Spring3引入了Spring表达式语言(Spring Expression Language,SpEL),它能够以一种强大和简介的方式将值装配到bean属性和构造器参数中。SpEL拥有很多特性,包括:

  • a、使用bean的ID来引用bean;

  • b、调用方法和访问对象的属性;

  • c、对值进行算数、关系和逻辑运算;

  • d、正则表达式匹配;

  • e、集合操作。

SpEL不仅能够用在依赖注入中,在其他的一些方面也有很大用处。如Spring Security支持使用SpEL表达式定义安全限制工作。在Spring MVC应用中使用Thymeleaf模板作为视图的话,那么这些模板可以使用SpEL表达式引用模型数据。
SpEL表达式要放到”#{…}”之中,这与属性占位符有些类似,属性占位符需要放到”${…}”之中。下面是几个简单的SpEL样例,这些例子可以我们帮助编写自己的表达式。
#{1}:表达式中只有字面量1,当然它的计算结果就是1.
#{T(System).currentTimeMillis()}:它的最终结果是计算表达式的那一刻当前时间的毫秒数。T()表达式会将java.lang.System视为Java中对应的类,因此可以调用其static修饰的currentTimeMillis()方法。
#{sgtPeppers.artist}:引用ID为sgtPeppers的bean的artist属性。
#{systemProperties[‘disc.title’]}:通过systemProperties对象引用系统属性。

在bean装配时使用SpEL表达式

如果通过组件扫描创建bean的话,在注入属性和构造器参数时,我们可以使用@Value注解,这与之前看到的属性占位符非常类似。不过,我们将用SpEL表达式替代属性占位符。例如,下面的样例展现了BlankDisc,它会从系统属性中获取专辑名称和艺术家的名字:

public BlankDisc(@Value("#{systemProperties['disc.title']}") String title,
                 @Value("#{systemProperties['disc.artist']}") String artist){
    this.title=title;
    this.artist=artist;
}

在XML配置中,你可以将SpEL表达式传入或的value属性中,或者将其作为p-命名空间或c-命名空间条目的值。例如,在如下BlankDisc bean的XML声明中,构造器参数就是通过SpEL表达式设置的:

systemProperties['disc.title']}"
      c:_artist="#{systemProperties['disc.artist']}" />

上面已经看过了几个简单的样例,也学习了如何将SpEL解析得到的值注入到bean中,下面将是SpEL所支持的基础表达式学习。

表示字面值

SpEL表达式可以用来表示各种类型的字面量,如:正数:#{1}、浮点数:#{3.14159}、String:#{‘Hello’}、布尔型:#{true},用SpEL表示字面量并没有太大用处,但是要记住一点,更有意思的SpEL表达式是由更简单的表达式组成的,因此了解在SpEL中如何使用字面量还是很有用处的。但组合更为复杂的表达式时,你迟早会用到它们。

引用bean、属性和方法

SpEL能做的另一件基础的侍寝就是通过ID引用其他的bean。例如,你可以使用SpEL将一个bean装配到另一个bean的属性中,此时要使用bean ID作为SpEL表达式(在本例中,也就是sgtPeppers):

#{sgtPeppers}

现在,假设我们想在一个表达式中引用sgtPeppers的artist属性:

#{sgtPeppers.artist}

表达式主体的第一部分引用了一个ID为sgtPeppers的bean,分隔符之后是对artist属性的引用。除了引用属性之外,我们还可以调用bean上的方法。例如,假设有另外一个bean,它的ID为artistSelector,我们可以在SpEL表达式中按照如下的方式来调用bean的selectArtist()方法:

#{artistSelector.selectArtist()}

对于被调用方法的返回值来说,我们同样可以调用它的方法。如果selectArtist()方法返回的是一个String,那么可以调用toUpperCase()将整个艺术家的名字改为大写字母形式:

#{artistSelector.selectArtist().toUpperCase()}

如果selectArtist()的返回值不是null的话,这没有什么问题。为了避免出现NullPointerException,我们可以使用类型安全的运算符:

#{artistSelector.selectArtist()?.toUpperCase()}

与之前只是使用点号(.)来访问toUpperCase()方法不同,现在我们使用了“?.”运算符。这个运算符能够在访问它右边的内容之前,确保它对应的元素不是null。所以,如果selectArtist()的返回值是null的话,那么SpEL将不会调用toUpperCase()方法。表达式的返回值会是null。

在表达式中使用类型

如果要在SpEL中访问类作用域的方法和常量的话,要依赖T()这个关键的运算符。例如,为了在SpEL中表达Java的Math类,需要按照如下的方式使用T()运算符:

#{T(java.lang.Math).random}     //引用Math类的随机数方法
#{T(java.lang.Math).PI}    //引用Math类的常量

SpEL运算符

SpEL提供了多个运算符,这些运算符可以用在SpEL表达式的值上。下表为用来操作表达式值的SpEL运算符:
运算符类型 运算符
算数运算 +、-、*、/、%、^
比较运算 <、>、==、<=、>=、lt、gt、eq、le、ge
逻辑运算 and、or、not、|
条件运算 ?:(ternary)、?:(Elvis)
正则表达式 matches
下面这个例子作为使用上述运算符的一个简单样例:

#{2*T(java.lang.Math).PI*circle.radius}

在这里PI的值乘以2,然后再乘以radius属性的值,这个属性来源于ID为circle的bean。实际上它计算了圆的周长。
当使用String类型的值时,“+”运算符执行的是连接操作,与在Java中是一样的:

#{disc.title+'by'+disc.artist}

SpEL同时还提供了比较运算符,用来在表达式中对值进行比较。注意在上表中,比较运算符有两种形式:符号形式和文本形式。例如:

#{counter.total==100}
//等同于下面的表达式
#{counter.total eq 100}

SpEL还提供了三元运算符(ternary),它与Java中的三元运算符非常类似。例如,如下的表达式会判断如果scoreboard.score>1000的话,计算结果为String类型的“Winner!”,否则,结果为“Loser”:

#{scoreboard.score>1000 ? "Winner!":"Loser"}

三元运算符的一个常见场景就是检查null值,并用一个默认值来代替null。例如,如下的表达式会判断disc.title的值是不是null,如果是null的话,那么表达式的计算结果就会是“The Value Is Null”:

#{disc.title ?: 'The Value Is Null'}

这种表达式通常称为Elvis运算符。这个奇怪名称的来历是,当使用符号来表示表情时,问号看起来很像是猫王(Elvis Presley)的头发。

计算正则表达式

当处理文本时,又是检查文本是否匹配某种模式是非常有用的。SpEL通过matches运算符支持表达式中的模式匹配。matches运算符对String类型的文本(作为左边参数)应用正则表达式(作为右边参数)。matches的运算结果会返回一个Boolean类型的值:如果与正则表达式相匹配,则返回true,否则返回false。加入我们想判断一个字符是否包含有效的邮件地址。在这个场景下,我们可以使用matches运算符,如下所示:

#{admin.email matches '[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-+\\.com]'}

计算集合

SpEL中最令人旌旗的一些技巧是与集合和数组有关的。最简单的事情可能就是引用列表中的一个元素了。

#{jukebox.songs[4].title}

这个表达式会计算songs集合汇总第五个元素的title属性,这个集合来源于ID为jukebox bean。
SpEL还提供了查询运算符(.?[]),它会用来对集合进行过滤,得到集合的一个子集。作为阐述的样例,假设你希望得到jukebox中artist属性为Aerosmith的所有歌曲。如下的表单时就使用查询运算符得到了结果:

#{jukebox.songs.?[artist eq 'Aeronsmith']}

可以看到,选择运算符在它的方括号中接收另一个表达式。SpEL还提供了另外两个查询运算符:”.^[]“和”.$[]“,它们分别用来在集合中查询第一个匹配项和最后一个匹配项。例如,下面的表达式会查找列表中第一个artist属性为Aerosmith的歌曲:

#{jukebox.songs.^[artist eq 'Aerosmith']}

最后SpEL还提供了投影运算符(.![]),它会从集合的每个成员中选择特定的属性放到另外一个集合中。作为样例,假设我们不想要歌曲对象的集合,而是所有歌曲名称的集合。如下的表达式会将title属性投影到一个新的String类型的集合中:

#{jukebox.songs.![title]}

实际上,投影操作可以与其他任意的SpEL运算符一起使用。比如,我们可以使用如下的表达式获得Aerosmith所有歌曲的名称列表:

#{jukebox.songs.?[artist eq 'Aerosmith'].![title]}

我们所介绍的只是SpEL功能的一个皮毛,但已经能够让我们对SpEL有一个大致的了解了,当需要用到的时候再去深入学习。在动态注入值到Spring Bean时,SpEL是一种很便利和强大的方式。我们又时会忍不住编写很复杂的表达式。但需要注意的是,不要让你的表达式太智能了。你的表达式越智能,对它的测试就越重要。SpEL毕竟只是String类型的值,可能测试起来很困难。鉴于这一点,我建议尽可能让表达式保持简洁,这样测试不会是什么大问题。

你可能感兴趣的:(WEB——Spring)