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 条评论
登录 后参与评论

相关文章

来自专栏屈定‘s Blog

设计模式--Builder模式的思考

在日常开发中总是会遇到多参数的情况,那么对于多参数,尤其是可选参数众多的情况,可能有如下的一些解决方案.

1539
来自专栏ml

jsp基础知识(基本的语法及原理)

     jsp 语法分为三种不同的类型: (1)  编译器指令: 类如: <%@ page import="java.io.*"%> (2)  脚本语法:  ...

2704
来自专栏后端技术探索

nginx location配置(转)

今天讲下location的用法,部分内容是直接从网络上摘取的,这边做了一个整理,为了便于理解和学习,我这边做了一些例子。

884
来自专栏闪电gogogo的专栏

Python初学——多进程Multiprocessing

1.1 什么是 Multiprocessing 多线程在同一时间只能处理一个任务。 可把任务平均分配给每个核,而每个核具有自己的运算空间。 1.2 添加进程 P...

2618
来自专栏恰同学骚年

Entity Framework 基础知识走马观花

  (1)通过选择以XML方式打开edmx文件,我们可以可以清楚地看到,edmx模型文件本质就是一个XML文件;

862
来自专栏代码世界

Django之路由系统

Django的路由系统   URL配置其实就是告诉Django项目你执行代码的路径,本质就是路径和调用的视图函数之间的映射关系表。Django通过这个表,可以把...

2757
来自专栏我爱编程

Day20内建模块itertools&urllib

itertools Python的内建模块itertools提供了非常有用的用于操作迭代对象的函数。 首先,我们看看itertools提供的几个“无限”迭代器...

3299
来自专栏javathings

volatile 解决了什么问题?

volatile 关键字是最常问到的问题,关于这个关键字的作用解释,网上的文章已经多如牛毛了。

1113
来自专栏抠抠空间

django之urls系统

Django的urls系统简介 Django 1.11版本 URLConf官方文档:https://docs.djangoproject.com/en/1.11...

2567
来自专栏技术墨客

Nginx域名访问处理过程 原

在实际应用中,我们可以将多个域名指向一个IP 地址,或者使用范IP解析功能。当多个域名执行一个 IP 地址时,Nginx 可以根据域名来分配不同的虚拟服务器,如...

823

扫码关注云+社区