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

1.1注入外部的值

Spring中,处理外部值的最简单方式就是声明属性源并通过Spring的Environment来检索属性。例如,程序清单3.7展现了一个基本的Spring配置类,它使用外部的属性来装配BlankDisc bean。

程序清单3.7 使用@PropertySource注解和Environment

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

在本例中,@PropertySource引用了类路径中一个名为app.properties的文件。它大致会如下所示:

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

这个属性文件会加载到Spring的Environment中,稍后可以从这里检索属性。同时,在disc()方法中,会创建一个新的BlankDisc,它的构造器参数是从属性文件中获取的,而这是通过调

用getProperty()实现的。

当我们去了解Environment的时候会发现,程序清单3.7所示的getProperty()方法并不是获取属性值的唯一方法,getProperty()方法有四个重载的变种形式

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

前两种形式的getProperty()方法都会返回String类型的值。我们已经在程序清单3.7中看到了如何使用第一种getProperty()方法。但是,你可以稍微对@Bean方法进行一下修改,这样在指定属性不存

在的时候,会使用一个默认值:

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

剩下的两种getProperty()方法与前面的两种非常类似,但是它们不会将所有的值都视为String类型。例如,假设你想要获取的值所代表的含义是连接池中所维持的连接数量。如果我们从属性文件中得到的是一个String类型的值,那么在使用之前还需要将其转换为Integer类型。但是,如果使用重载形式的getProperty()的话,就能非常便利地解决这个问题:

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

直接从Environment中检索属性是非常方便的,尤其是在Java配置中装配bean的时候。但是,Spring也提供了通过占位符装配属性的方法,这些占位符的值会来源于一个属性源。

解析属性占位符

Spring一直支持将属性定义到外部的属性的文件中,并使用占位符值将其插入到Spring bean中。在Spring装配中,占位符的形式为使用“${... }”包装的属性名称。作为样例,我们可以在XML中按照如下的

方式解析BlankDisc构造器参数:

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

可以看到,title构造器参数所给定的值是从一个属性中解析得到的,这个属性的名称为disc.title。artist参数装配的是名为disc.artist的属性值。按照这种方式,XML配置没有使用任何硬编码的值,它的值是从配置文件以外的一个源中解析得到的。(我们稍后会讨论这些属性是如何解析的。)

如果我们依赖于组件扫描和自动装配来创建和初始化应用组件的话,那么就没有指定占位符的配置文件或类了。在这种情况下,我们可以使用@Value注解,它的使用方式与@Autowired注解非常相似。比

如,在BlankDisc类中,构造器可以改成如下所示:

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

为了使用占位符,我们必须要配置一个PropertyPlaceholderConfigurer bean或PropertySourcesPlaceholderConfigurer bean。从Spring3.1开始,推荐使用PropertySourcesPlaceholderConfigurer,因为它能够基于Spring Environment及其属性源来解析占位符。如下的@Bean方法在Java中配置了PropertySourcesPlaceholderConfigurer:

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

如果你想使用XML配置的话,Spring context命名空间中的<context:propertyplaceholder>元素将会为你生

成PropertySourcesPlaceholderConfigurer bean:

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

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

Spring 3引入了Spring表达式语言(Spring Expression Language,SpEL),它能够以一种强大和简洁的方式将值装配到bean属性和构造器参数中,在这个过程中所使用的表达式会在运行时计算得到值。使用SpEL,你可以实现超乎想象的装配效果,这是使用其他的装配技术难以做到的(甚至是不可能的)。SpEL拥有很多特性,包括:

使用bean的ID来引用bean;

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

对值进行算术、关系和逻辑运算;

正则表达式匹配;
集合操作。

在本书后面的内容中你可以看到,SpEL能够用在依赖注入以外的其他地方。例如,Spring Security支持使用SpEL表达式定义安全限制规则。另外,如果你在Spring MVC应用中使用Thymeleaf模板作为视图的话,那么这些模板可以使用SpEL表达式引用模型数据。作为起步,我们看几个SpEL表达式的样例,以及如何将其注入到bean中。然后我们会深入学习一些SpEL的基础表达式,它们能够组合起

来形成更为强大的表达式。

需要了解的第一件事情就是SpEL表达式要放到“#{ ... }”之中,这与属性占位符有些类似,属性占位符需要放到“${ ... }”之中。下面所展现的可能是最简单的SpEL表达式了:

当然,在实际的应用程序中,我们可能并不会使用这么简单的表达式。我们可能会使用更加有意思的表达式,如:

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

它的最终结果是计算表达式的那一刻当前时间的毫秒数。T()表达式会将java.lang.System视为Java中对应的类型,因此可以调用其static修饰的currentTimeMillis()方法。

SpEL表达式也可以引用其他的bean或其他bean的属性。例如,如下的表达式会计算得到ID为sgtPeppers的bean的artist属性

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

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

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

引用bean、属性和方法

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

是sgtPeppers):

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

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

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

除了引用bean的属性,我们还可以调用bean上的方法。例如,假设有另外一个bean,它的ID为artistSelector,我们可以在SpEL表达式中按照如下的方式来调用bean的selectArtist()方法:

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

在表达式中使用类型

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

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

这里所示的T()运算符的结果会是一个Class对象,代表了java.lang.Math。如果需要的话,我们甚至可以将其装配到一个Class类型的bean属性中。但是T()运算符的真正价值在于它能够

访问目标类型的静态方法和常量。

有关spel的表达式这里就不详细介绍了,可以参看Spring实战第三章,spring高级装配章节!

上一篇:Spring表达式语言SpEL


下一篇:开涛spring3(5.1&5.2) - Spring表达式语言 之 5.1 概述 5.2 SpEL基础