Spring框架学习之高级依赖关系配置(二)

     紧接着上篇内容,本篇文章将主要介绍XML Schema的简化配置和使用SpEL表达式语言来优化我们的配置文件。

一、基于XML Schema的简化配置方式 从Spring2.0以来,Spring支持使用XML Schema来简化配置。在以前的bean元素配置下,所有的属性注入都需要一个property元素,集合属性就需要更多的这样的元素,一旦项目庞大,整个配置文件将无法维护。XML Schema提供的命名空间,可以帮助我们的配置文件缩减容量。主要有三种命名空间:

  • p:命名空间简化属性配置
  • c:命名空间简化构造配置
  • util:命名空间简化集合配置

1、使用p:命名空间简化配置 首先看一个我们常见的bean的配置:

<bean id="person" class="MyPackage.Person">
    <property name="name" value="single"/>
    <property name="age" value="22"/>
</bean>

这是一个再普通不过的bean的配置,它等效于:

<bean id="person" class="MyPackage.Person" p:name="single" p:age="22" />

但是这么做的前提是,在Spring配置文件的XML头部导入p的命名空间:

xmlns:p="http://www.springframework.org/schema/p"

无论你是使用IDE自动导入的,还是使用工具,或者自己手动导入的,想要使用p Schema,就必须该命名空间。p是property的缩写,除了可以注入普通的值类型之外,还可以注入引用类型,例如:

<bean id="person" class="MyPackage.Person" p:name-ref="stuName" p:age="22" />

2、使用c:命名空间简化配置 我们使用p命名空间简化了设置注入操作,c命名空间则是constructor的缩写,它用于简化构造注入的操作。例如:

<bean id="person" class="MyPackage.Person" c:name="single" c:age="22" />

Spring最后会调用构造注入,将name和age作为参数出入bean类的构造器中,构造实例返回。除此之外,还可以这么写:

<bean id="person" class="MyPackage.Person" c:_0="single" c:_1="22" />

下划线加上数字表示,这是构造器中的第几个参数,到时候会被Spring按序传入构造器中。

3、使用util命名空间简化配置 util给我们提供了以下几个元素:

  • constant:对于在配置文件中获取指定类的静态Field的值的一个简化配置
  • property-path:对于在配置文件中获取调用getter方法的一个简化配置
  • list:简化list作为bean的配置
  • map:简化map作为bean的配置
  • set:简化set作为bean的配置
  • properties:加载一份properties属性文件

^1、constant 首先看一段我们之前用于获取一个类的静态属性的配置:

<bean id="person" class="MyPackage.Person">
    <property name="name" value="single"/>
    <property name="age">
        <bean class="org.springframework.beans.factory.config.FieldRetrievingFactoryBean">
            <property name="targetClass" value="MyPackage.BeanStaticClass"/>
            <property name="targetField" value="age"/>
        </bean>
    </property>
</bean>

这是一段我们之前用于在配置文件中获取类静态属性的代码,总体上来说还是挺繁琐的,下面我们使用util下的constant来简化这段配置。

<bean id="person" class="MyPackage.Person">
    <property name="name" value="single"/>
    <property name="age">
        <util:constant static-field="MyPackage.BeanStaticClass.age"/>
    </property>
</bean>

static-field属性的值指定了需要访问那个类下的那个静态属性。如果需要将该值单独定义在容器中,可以为其增加id属性以便其他bean实例获取。

^2、property-path 对于在配置文件中直接调用其他bean的getter方法,我们一般如是配置:

<bean id="student" class="MyPackage.Student">
    <property name="age" value="22"/>
</bean>

<bean id="person" class="MyPackage.Person">
    <property name="name" value="single"/>
    <property name="age">
        <bean class="org.springframework.beans.factory.config.PropertyPathFactoryBean">
            <property name="targetBeanName" value="student"/>
            <property name="propertyPath" value="age"/>
        </bean>
    </property>
</bean>

一般我们需要这么写来配置,下面我们通过util的property-path来简化配置:

<bean id="student" class="MyPackage.Student">
    <property name="age" value="22"/>
</bean>

<bean id="person" class="MyPackage.Person">
    <property name="name" value="single"/>
    <property name="age">
        <util:property-path path="student.age"/>
    </property>
</bean>

property-path的值指定了需要调用那个bean实例的那个getter方法。如果需要单独将该值配置在容器中,可以为其指定id。

^3、list 通过util,Spring允许我们单独定义一个list bean在容器中。一个list bean给我们提供如下几个属性:

  • id:容器中的唯一标识
  • list-class:指定实现该list的实现类,默认为ArrayList
  • scope:指定该list的作用域

下面我们定义一个list bean:

<util:list id="list" list-class="java.util.ArrayList">
    <value>nanjing</value>
    <value>nantong</value>
    <value>yancheng</value>
</util:list>

^4、map和set 有关map和set的util使用,基本和list的使用情况类似,此处只给出创建bean的配置代码,其他的不再赘述。

//map
<util:map id="map" map-class="java.util.HashMap">
    <entry key="中国银行" value="241241212" />
   <entry key="建设银行" value="543534545"/>
</util:map>
//set
<util:set id="set" set-class="java.util.HashSet">
    <value>single</value>
    <value>cyy</value>
</util:set>

至此,有关XML Schema的基本内容已如上述所述,Spring中还有一些其他相关的Schema文件,它们各自有各自的作用,有用于简化AOP配置的,有用于简化事务配置的等等,我们将在后续文章中进行学习。

二、使用Spring的EL表达式语言 Spring的配置文件有一个非常明显的缺陷,大量的静态注入,动态性不强。而SpEL类似于jsp的EL,采用表达式的方式为属性注入值,当程序动态运行时,这些表达式的值才确定。SpEL可以单独使用,也可以在Spring配置文件中使用,我们此处主要介绍在配置文件中的SpEL的使用。

1、创建数组 有时我们的实例中有类型为数组的属性,那么我们就可以通过SpEL定义数组作为参数注入依赖给该属性。

<bean id="person" class="MyPackage.Person">
    <property name="array" value="#{new int[]{12,2,5,7,8,6,8}}" />
</bean>

在Spring配置中使用SpEL中,基本的使用格式如下:

#{expression}

2、创建list集合 我们也可以使用SpEL定义list集合,例如:

<bean id="person" class="MyPackage.Person">
    <property name="list" value="#{{'single','cyy','xijingping'}}"/>
</bean>

list集合的定义使用 "{....}",大括号中的每个元素都对应于list中的一个元素。

我们可以通过以下语法格式访问list中的元素:

listName[index]

访问容器中map集合中的元素:

mapName[key]

例如:

<bean id="person" class="MyPackage.Person">
    <property name="name" value="#{list[0]}"/>
</bean>

这段配置会向person实例中的name属性注入容器中已经定义好的名为list的第一个元素的值。

3、调用方法 在SpEL中调用任意方法将会比我们之前介绍的那种纯配置形式简单很多,例如:

<bean id="person" class="MyPackage.Person">
    <property name="name" value="#{'hello'.concat(' world!')}"/>
</bean>

我们这里调用了jdk中的方法为person实例的name属性注入依赖值,当然也可以是我们自己定义的方法,可以是任意的方法。

4、类运算符 SpEL提供了一种运算符:T()。该运算符告诉Spring将括号中的内容作为一类类型而不是作为字符串进行解析。例如:

<bean id="person" class="MyPackage.Person">
    <property name="name" value="#{java.lang.Math.random()}"/>
</bean>

这样书写,Java会给我们报错,说java.lang.Math.random()并不是一个字符串,但是如果给他加上单引号,那么该函数就会以纯字符串作为参数注入给name属性,并不会被识别为正常函数。

正确书写格式如下:

<bean id="person" class="MyPackage.Person">
    <property name="name" value="#{T(java.lang.Math).random()}"/>
</bean>

5、集合选择 使用SpEL集合选择的基本格式如下:

collection.?[condition_expr]

只有符合条件condition_expr的集合元素才会被选择出来,例如:

<bean id="person" class="MyPackage.Person">
    <property name="list" value="#{list.?[length()>7]}"/>
</bean>

Spring会遍历整个list,每个元素都会调用他的length方法判断是否大于7,如果不是将舍弃该元素,否则该元素才会被帅选出来。

6、集合投影 集合投影可以让集合中的每个元素都去执行同一个方法,每个元素调用同一个方法,不同元素的该方法的返回值将构成最终的新集合。例如:

<bean id="person" class="MyPackage.Person">
    <property name="list" value="#{list.![length()]}"/>
</bean>

最终新集合中的元素是原集合中每个元素的长度。

SpEL的用法远非如此,这里只是列举了常用的几种用法。

两篇文章加强了对bean配置的理解,有总结不到之处,望指出!

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏帅小子的日常

spring的简单入门

3627
来自专栏乐百川的学习频道

Powershell快速入门(二) Shell编程

这一部分着重于介绍Powershell的程序知识,让我们能够编写功能强大的Powershell脚本,执行比较复杂的任务。 变量 变量使用$变量名创建和引用。举个...

22810
来自专栏Golang语言社区

编写地道的Go代码

最地道的Go代码就是Go的标准库的代码,有空的时候可以多看看Google的工程师是如何实现的。 1. 注释 可以通过/* ... */或者//增加注释, //之...

2716
来自专栏青枫的专栏

day25_day27_Struts2_学习回顾

        表现层、MVC模式。 2、Struts1和Struts2的一个显著区别是什么?     答:

815
来自专栏LanceToBigData

struts2(三)之表单参数自动封装与参数类型自动转换

前言   对struts2的使用不外乎这几点,参数自动封装,拦截器的使用,数据校验,ognl表达(值栈和actionContext的讲解),struts2的标签...

20410
来自专栏MasiMaro 的技术博文

VC 在调用main函数之前的操作

title: VC 在调用main函数之前的操作 tags: [VC++, 反汇编, C++实现原理] date: 2018-09-16 10:36:23 ...

682
来自专栏DOTNET

【翻译】MongoDB指南/CRUD操作(一)

【原文地址】https://docs.mongodb.com/manual/ MongoDB CRUD操作(一) 主要内容:CRUD操作简介,插入文档,查询文档...

2829
来自专栏青枫的专栏

java基础学习_IO流04_用户登录注册案例(IO版)、数据操作流(操作基本数据类型的流)、内存操作流、打印流、标准输入输出流、随机访问流、合并流、序列化流(对象操作流)、Properties属性集

java基础学习_IO流04_用户登录注册案例(IO版)、数据操作流(操作基本数据类型的流)、内存操作流、打印流、标准输入输出流、随机访问流、合并流、序列化流(...

401
来自专栏逆向技术

16汇编第十讲完结Call变为函数以及指令的最后讲解

16汇编完结Call变为函数以及指令的最后讲解 学了10天的16位汇编,这一讲就结束了,这里总结一下昨天的LOOP指令的缺陷,因为lOOP指令的缺陷,所以我们都...

15810
来自专栏互扯程序

你现在还在手动生成set,get方法吗?

JAVA面向对象编程中的封闭性和安全性。封闭性即对类中的域变量进行封闭操作,即用private来修饰他们,如此一来其他类则不能对该变量访问。这样我们就将...

1636

扫码关注云+社区