spring的高级装配之运行时值注入

746 阅读8分钟

运行时值注入

通常所讨论的是将一个bean引用注入到另一个bean的属性或构造器参数中。它通常来讲指的是将一个对象与另一个对象进行关联。
但是bean装配的另外一个方面指的是将一个值注入到bean的属性或者构造器参数中。
如将专辑的名字 装配到BlankDisc bean的构造器或title属性中:
1.

以上两种方式都是硬编码,有时候硬编码是可以的,但有的时候,我们可能会希望避免硬编码值,而是想让这些值在运行时再确定。为了实现这些功能,Spring提供了两 种在运行时求值的方式:

  • 属性占位符(Property placeholder)。
  • Spring表达式语言(SpEL)。

注入外部的值

在Spring中,处理外部值的最简单方式就是:先声明属性源,然后通过Spring的Environment来检索属性。
下图展示的spring配置类使用外部的属性来装配BlankDisc bean。


@PropertySource引用了类路径中一个名为app.properties的文件:


这个属性文件会加载到Spring的Environment中,稍后可以从这里检索属性。

深入学习Spring的Environment
1.getProperty()方法的四种重载形式:

2.希望获取的属性必须定义
如果你在使用getProperty()方法的时候没有指定默认值,并且这个属性没有定义的话, 获取到的值是null。但是如果希望这个属性必须定义的话,使用getRequiredProperty()方法
在这里,如果disc.title或disc.artist属性没有定义的话,将会抛出IllegalStateException异常。
3.如果想检查一下某个属性是否存在的话,那么可以调用Environment的containsProperty()方法:

4.如果想将属性解析为类的话,可以使用getPropertyAsClass()方法:

5.除了属性相关的功能以外,Environment还提供了一些方法来检查哪些profile处于激活状态:

  • String[] getActiveProfiles():返回激活profile名称的数组;
  • String[] getDefaultProfiles():返回默认profile名称的数组;
  • boolean acceptsProfiles(String...profiles):如果environment支持给定profile的话,就返回true。-使用acceptsProfiles()方法来确保给定bean所需的profile处于激活状态

解析属性占位符
Spring一直支持将属性定义到外部的属性的文件中,并使用占位符值将其插入到Spring bean中。在Spring装配中,占位符的形式为使用“${...}”包装的属性名称
1.在xml中

按照这种方式,XML配置没有使用任何硬编码的值,它的值是从配置文件以外的一个源中解析得到的。

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

使用占位符必须
我们必须要配置一个PropertyPlaceholderConfigurer bean或PropertySourcesPlaceholderConfigurer bean。从Spring 3.1开始,推荐使用PropertySourcesPlaceholderConfigurer,因为它能够基于Spring Environment及其属性源来解析占位符

  • 在javaConfig配置类中配置PropertySourcesPlaceholderConfigurer:
  • 想使用XML配置的话,Springcontext命名空间的<context:propertyplaceholder元素将会为你生 成PropertySourcesPlaceholderConfigurer bean:

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

Spring 3引入了Spring表达式语言(Spring Expression Language,SpEL),它能够以一种强大和简洁的方式将值装配到bean属性和构造器参 数中,在这个过程中所使用的表达式会在运行时计算得到值。

SpEL拥有很多特性,包括:

  • 使用bean的ID来引用bean;
  • 调用方法和访问对象的属性;
  • 对值进行算术、关系和逻辑运算; 正则表达式匹配;
  • 集合操作。

SpEL使用场景:
-依赖注入

  • SpEL能够用在依赖注入以外的其他地方。例如,SpringSecurity支持使用SpEL表达式定义安全限制规则。
  • 另外,如果你在SpringMVC应用中使用Thymeleaf模板作为视图的话,那么这些模板可以使用SpEL表达式引用模型数据

SpEL样例
SpEL表达式要放到“#{ ... }”之中

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

3.可以通过systemProperties对象引用系统属性:
4.

  • 如果通过组件扫描创建bean的话,在注入属性和构造器参数时,我们可以使用@Value注解,这与之前看到的属性占位符非常类似。不过,在 这里我们所使用的不是占位符表达式,而是SpEL表达式。

  • 在XML配置中,你可以将SpEL表达式传入或的value属性中,或者将其作为p-命名空间或c-命名空间条 目的值。

SpEL所支持的基础表达式

  • 表示字面值
    使用SpEL可以用来表示整数字面量、浮点数、String值以及Boolean值。 1.SpEL表达式表示浮点数、字符串、boolean值


  • 引用bean、属性和方法
    SpEL所能做的另外一件基础的事情就是通过ID引用其他的bean
    1)使用SpEL将一个bean装配到另外一个bean的属性中,此时要 使用bean ID作为SpEL表达式


    2)一个表达式中引用sgtPeppers的artist属性:
    表达式主体的第一部分引用了一个ID为sgtPeppers的bean,分割符之后是对artist属性的引用。
    3)1、除了引用bean的属性,我们还可以调用bean上的方法

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

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

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

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

    1.这里所示的T()运算符的结果会是一个Class对象,代表了java.lang.Math。
    2.T()运算符的真正价值在于它能够访问目标类型的静态方法和常量:

  • SpEL运算符
    SpEL提供了多个运算符,这些运算符可以用在SpEL表达式的值上:


样例:
1.

在这里PI的值乘以2,然后再乘以radius属性的值,这个属性来源于ID为circle的bean。实际上,它计算了circle bean中所定义圆的周长。

2.当使用String类型的值时,“+”运算符执行的是连接操作,与在Java中是一样的:


3.三元运算符
SpEL还提供了三元运算符(ternary),它与Java中的三元运算符非常类似

1)例如,如下的表达式会判断如果scoreboard.score>1000的 话,计算结果为String类型的“Winner!”,否则的话,结果为Loser:


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

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

  • 计算集合
    SpEL中最令人惊奇的一些技巧是与集合和数组相关的。最简单的事情可能就是引用列表中的一个元素了:

    这个表达式会计算songs集合中第五个(基于零开始)元素的title属性,这个集合来源于ID为jukebox bean。
    1、“[]”运算符用来从集合或数组中按照索引获取元素
    实际上,它还可以从String中获取一个字符。比如:
    这个表达式引用了String中的第四个(基于零开始)字符,也就是“s”。
     2、
    1)SpEL还提供了查询运算符(.?[]),它会用来对集合进行过滤,得到集合的一个子集。
    作为阐述的样例,假设你希望得到jukebox 中artist属性为Aerosmith的所有歌曲。如下的表达式就使用查询运算符得到了Aerosmith的所有歌曲:

         2)


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