这几天笔者阅读了一下Spring In Action 4,感觉还是有一定的收获的。在之前的项目上,只会简单地使用Spring MVC,对于很多概念并不理解。现在看看Spring的一些概念,如DI和AOP,以及Spring的各个模块,对于Spring这一整体框架可以有较深的了解。
这篇文章将主要以翻译的形式来讲解Spring,中间夹杂对于原文的理解,以及代码实现,相信对于读者会带来一定的帮助,也是对自己阅读的一个总结。如果读者有不同的见解,还望可以共同探讨一下。顺带推荐一下Manning的In Action系列,笔者此前读了《机器学习实战》,外加上现在看的《Spring实战》,感觉真心不错,国外的许多书籍,相比国内简直是良心之作。但是,中文译本的更新永远无法赶上英文原本的进度,就拿Spring来说,第三版的中文译本还停留在Spring 3,然而Spring已经升级到了4版本,而英文原本紧跟技术进度,相比之下,第四版的中文译本,还不知要到何年何月。当然,文章的翻译除了要花费长时间的努力,还需要译者对技术有较为深入的了解,在此,还是要感谢那些辛勤的贡献者。笔者翻译此书的目的,是为了在读书的同时,能够总结到一些有用的知识,并期望能给读者带来一定的收获。
(题外话:由于开通了邮箱通知功能,近期Windows10任务栏右边总是会显示邮件提示。有一些问题,由于笔者也没有去深度地探究Spring MVC这个框架,也无法做出具体的回答,但相信前面几篇文章对于入门开发还是存在一定的帮助的。真正的项目开发肯定会比教程中更加复杂,因为除了业务逻辑,还有许多重要的问题,如高并发、多线程,以及至关重要的安全问题,是需要考虑进来的。)
接下来,进入正题。。。(转载请说明出处:Gaussic,AI研究生)
Spring能做很多事情。但是对于Spring的所有的奇妙功能,它们的主要特征是依赖注入(dependency indection, DI)和面向方面编程(aspect-oriented programming, AOP,也可以译为面向切面编程)。
第一章,付诸行动(Springing into action),我将带给你一个Spring 框架的快速概览,包括Spring中的DI和AOP的快速概览,并展示它们是如何帮助降低应用组件(components)的耦合度的。
第二章,装配Bean(Wiring Beans),我们将更为深入地探讨如何将应用的各个组件拼凑在一起。我们将着眼于自动配置(automatic configuration)、基于Java的配置,和XML配置,这三种Spring提供的配置方法。
第三章,高级装配(Advanced Wiring),将展示一些小把戏和手段,它们将帮助你深度挖掘Spring的强大能力,包括有条件的配置(conditional configuration),用以处理自动装配(autowiring)、范围控制(scoping)以及Spring表达式语言(Spring Expression Language, Spring EL)。
第四章,面向方面的Spring(Aspect-oriented Spring),探究如何使用Spring的AOP特征来解除系统级的服务(例如安全和审计(auditing))以及它们所服务的对象之间的耦合度。这一章为之后的章节搭建了一个起始平台,例如第9、13和14章,它们讲解了如何在声明式安全(declarative security)和缓存。
对于Java开发者来说,现在是一个很好的时期。
在20年的历史中,Java经历过一些好的时期以及不好的时期。尽管存在着一些粗糙点(rough spots),例如applets, Enterprise JavaBeans(EJB), Java Data Objects(JDO),以及数不尽的日志框架,Java仍然拥有着一个丰富的多样化的历史,作为许多企业级软件的平台而言。而且,Spring在这一段历史上占有很重要的一笔。
在早期,Spring作为更加重量级的Java技术(特别是EJB)的替代品被创造出来。相比EJB,Spring提供了一个更轻量更简洁的的编程模型。它给予了普通Java对象(POJO)更加强大的能力,而这些能力是EJB和一些其他Java规格才拥有的。
随着时间的过去,EJB和J2EE得到了发展。EJB开始自身提供简单的面向POJO的变成模型。现在EJB利用了诸如依赖注入和面向方面变成的思想,可以说其灵感来自于Spring的成功。
虽然J2EE(现在被称为JEE)可以赶上Spring,但是Spring从未停止前进的脚步。Spring持续的发展各个领域。而这些领域,即使是现在,JEE才刚刚开始开始探索,甚至还未引入。移动开发,社交API集成,NoSQL数据库,云计算,以及大数据只是Spring引入的一小部分领域。而且,未来的Spring依然一片光明。
正如我所说,对于开发者来说,现在是一个很好的时期。
这本书是对Spring的一个探索。在这一章节,我们将在高层次测试Spring,让你品尝一下Spring到底是什么。这一章将让你明白Spring到底解决了各种什么样的问题,并且未这本书的其余部分做好准备。
Spring是由Rod Johnson创造的一个开源框架。Spring被创造出来以解决企业级应用开发的复杂问题,并且让普通的JavaBeans(plain-vanilla JavaBeans)能够完成此前只有EJB才可能完成的任务。但是Spring的用处并不仅仅局限于服务器端开发。而且Java应用可以从Spring中获益,例如间接性、可测试性以及松耦合。
另一种意义的bean...虽然在表示应用组件时,Spring大方地使用了bean和JavaBean这两个词,但这并不表示一个Spring组件必须严格地遵从JavaBean的规格。一个Spring组件可以是任何形式的POJO。在本书中,我假设JavaBean是一个宽松的定义,与POJO同义。
通过本书你将了解到,Spring做了许多事情。但是Spring提供的几乎每一个功能的根基,是一些非常基础的想法,全部专注于Spring的基本任务:Spring简化Java开发。
这里是粗体!大量的框架说要简化某一些事物。但是Spring旨在简化Java开发这一广泛主题。这还需要更多的解释。Spring是如何简化Java开发的呢?
为了支持对Java复杂度的攻击,Spring利用了4个关键策略:
利用POJO的轻量级与最小侵入式开发(Lightweight and minimally invasive development with POJOs)
通过DI和面向接口实现松耦合(Loose coupling through DI and interface orientation)
通过方面和普通惯例实现声明式编程(Declarative programming through aspects and common conventions)
利用方面和模板消除陈词滥调的代码(Eliminating boilerplate code with aspects and templates)
基本上Spring做的每一件事都可以追溯到这四条策略中的一条或几条上。在这一章节的其他部分,我将扩展到这些想法的每一个中去,展示具体的例子以解释Spring是如何完成其简化Java开发的目标的。让我们从Spring如何通过鼓励面向POJO的开发来保持最小化侵入式的(minimally invasive,这点不好翻译,大致意思是不怎么需要修改POJO的代码)。
如果你做过很长时间的Java开发工作,你可能会发现某些框架会牢牢地困住你,它们强制你去继承某一个类或实现某一个接口。侵略式编程模型一个简单目标例子(easy-target example)是EJB 2-era stateless session beans(不详,在此不做翻译)。但是即使早期的EJBs是这样一个简单目标,侵入式编程编程仍然能在早期版本的Struts, WebWork, Tapestry,以及无数其他的Java规格与框架中找到。
Spring(尽可能地)避免其API污染你的应用代码。Spring几乎从来不强迫你去实现一个Spring接口,或集成一个Spring类。反而,一个基于Spring的应用中的类通常并没有指示它们为Spring所用。最坏情况,一个类可能被Spring的一个注解标注,但是它另一方面是一个POJO。
为了说明,考虑HelloWorldBean类。
如你所见,这很简单,就是一个普通的Java类,一个POJO。它并没有什么特殊的东西以表明它是一个Spring组件。Spring的非侵入式编程模型表示,这个类在Spring中同样可以很好的工作,就像它能在其他非Spring应用中一样。
尽管样式非常简单,POJOs依然可以很强大。Spring使得POJOs更强大的一个方法是利用DI装配(assembling)它们。让我们看看DI是如何帮助解除应用对象之间的耦合度的。
依赖注入这个词听起来可能有点高大上,让人联想出一些复杂的编程技术或设计模式的概念。但是结果证明,DI并没有听起来那么复杂。通过在你的项目中利用DI,你将发现你的代码将得到极大地简化,更容易理解,更容易测试。
任何非平凡的应用(比Hello World要复杂的多)都是有相互协作的多个类组成的,以执行一些业务逻辑。传统地,每一个对象负责获取对它自己的依赖(与他协作的对象)的引用。这将会带来高度耦合和难以测试的代码。
例如,考虑下面的Knight类。
package com.gaussic.knights; // 一个专门拯救少女的骑士 public class DamselRescuingKnight implements Knight { private RescueDamselQuest quest; // 拯救少女任务 // 这里出现了与RescueDamselQuest的紧密耦合 public DamselRescuingKnight() { this.quest = new RescueDamselQuest(); } // 骑士出征 public void embarkOnQuest() { quest.embark(); } }
相信大家都知道骑士与龙的故事,一个勇敢的骑士,除了能够打败巨龙之外,还需要拯救被囚禁的少女。但是在上面的DamselRescuingKnight中,我们发现他在构造器中创建了他自己的任务,一个RescueDamselQuest(也就是说,这个骑士构造出来时心里就只有一个任务,就是拯救少女,注意这个任务是发自内心的)。这导致了DamselRescuingKnight与RescueDamselQuest的高度耦合,这严重限制了骑士的行动。如果有一位少女需要拯救,那么这个骑士将会出现。但是如果需要屠龙呢,或者需要开个圆桌会议呢,那这个骑士将被一脚踹开。
此外,给DamselRescuingKnight写一个单元测试是非常困难的。在这个测试中,你需要能够保证,在骑士的embarkOnQuest()方法被调用时,quest的embark()方法也被调用。但是在这里并没有明确的方法来完成它。不幸的是,DamselRescuingKnight将保持未测试状态。
耦合是一个双头猛兽。一方面,紧密耦合的代码难以测试,难以重用,且难以理解,并且经常表现出“打地鼠”的bug行为(修复了一个bug,又产生了一个或多个新的bug)。另一方面,一定数量的耦合还是有必要的,完全不耦合的代码做不了任何事情。为了做一些有用的事情,类之间应该互相具有一定的认识。耦合是必需的,但是需要仔细管理。
利用DI,对象在构建时被给予它们的依赖,这通过一些定位系统中的每一个对象的三方来实现。对象不需要创建或获得它们的依赖。如图所示,依赖被注入到需要它们的对象中。
为了说明这一点,让我么看看BraveKnight类:一个骑士不仅仅要勇敢,还应该可以完成任何到来的任务。
package com.gaussic.knights; // 一个勇敢的骑士,应该能完成任何到来的任务 public class BraveKnight implements Knight { private Quest quest; // 任务,这是一个接口 // 构造器,注入Quest public BraveKnight(Quest quest) { this.quest = quest; } // 骑士出征 public void embarkOnQuest() { quest.embark(); } private BraveKnight() {} }
如你所见,BraveKnight不像DamselRescuingKnight一样创建自己的任务,而是在构造的时候被传入了一个任务作为构造参数(也就是说,这个骑士构造出来时,有人给了他一个任务,他的目标是完成这个任务,想雇佣兵一样)。这种类型的注入被称为构造注入(constructor injection)。
此外,勇敢的骑士所赋予的任务是一个Quest类型,它是一个接口,所有的具体quest都要实现这个接口。因而BraveKnight可以挑战RescueDamselQuest,SlayDragonQuest,MakeRoundTableRounderQuest,或者任何其他的Quest。
这里的关键点是,BraveKnight并不与任何一个特定的Quest的实现耦合。对于给予了什么任务,他并不介意,只要它实现了Quest接口。这就是DI的一个关键益处-松耦合。如果一个对象仅仅通过依赖的接口(而不是它们的实现或者实例化)来了解这些依赖,那么这个依赖就可以自由地更换为不同的实现,而一方(在此为BraveKnight)不需要知道有任何的不同。
换出依赖的一个最普遍的方法是在测试时利用一个mock实现(one of the most common ways a dependency is swapped out is with a mock implementation during test)。由于紧耦合,你无法适合地测试DamselRescuingKnight,但是你可以轻松地测试BraveKnight,通过给它一个Quest的mock实现(其实就是构建了一个虚拟的Quest),如下所示:
(注意:运行以下测试需要引入junit包和mockito-core包)
package com.gaussic.knights.test; import com.gaussic.knights.BraveKnight; import com.gaussic.knights.Quest; import org.junit.Test; import org.mockito.Mockito; // 一个勇敢的骑士,需要经历mock测试的考验 public class BraveKnightTest { @Test public void knightShouldEmbarkOnQuest() { Quest mockQuest = Mockito.mock(Quest.class); // 构建 mock Quest,其实就是虚拟的Quest BraveKnight knight = new BraveKnight(mockQuest); // 注入mockQuest knight.embarkOnQuest(); Mockito.verify(mockQuest, Mockito.times(1)).embark(); } }
在此使用了一个mock对象框架(Mockito)来创建一个Quest接口的虚拟实现。利用虚拟对象,我们可以创建一个新的BraveKnight实例,并通过构造器注入这一虚拟Quest。在调用embarkOnQuest()方法后,利用Mockito来验证虚拟Quest的embark()方法仅仅被调用一次。
现在,BraveKnight被写成了这样的形式,你可以将任何一个任务交给骑士,那么你应该如何指定给他什么Quest呢?假设,例如,你想要BraveKnight去完成一项屠龙的任务。可能是SlayDragonQuest,如下所示。
package com.gaussic.knights; import java.io.PrintStream; // 任务目标:屠杀巨龙 public class SlayDragonQuest implements Quest { private PrintStream stream; // 允许自定义打印流 public SlayDragonQuest(PrintStream stream) { this.stream = stream; } public SlayDragonQuest() {} // 任务说明 public void embark() { stream.println("Embark on quest to slay the dragon!"); } }
如你所见,SlayDragonQuest实现了Quest接口,让其能很好的适应BraveKnight。你也可能注意到,SlayDragonQuest在构造时请求了一个普通的PrintStream,而不像其他的简单Java示例一样依靠System.out.println()。这里有一个大问题,怎样才能把SlayDragonQuest交给BraveKnight呢?还有怎样才能把PrintStream交给SlayDragonQuest呢?
在应用组件之间建立联系的行为,通常被称之为装配(wiring)。在Spring中,有很多将组件装配在一起的方法,但是通常的方法总是通过XML的。下面展示了一个简单的Spring配置文件,knight.xml,它将BraveKnight, SlayDragonQuest和PrintStream装配在了一起。
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> <!-- BraveKnight bean,id为knight --> <bean id="knight" class="com.gaussic.knights.BraveKnight"> <!-- 构造参数,引用 id为quest的bean --> <constructor-arg ref="quest"/> </bean> <!-- SlayDragonQuest bean,id为quest --> <bean id="quest" class="com.gaussic.knights.SlayDragonQuest"> <!-- 构造参数,值为 #{T(System).out} --> <constructor-arg value="#{T(System).out}"/> </bean> </beans>
在此,BraveKnight和SlayDragonQuest在Spring中都被定义为bean。在BraveKnight bean中,它在构造是传入了一个SlayDragonQuest的引用,作为构造参数。同时,SlayDragonQuest bean中使用了Spring表达式语言来传递 System.out(这是一个PrintStream)给SlayDragonQuest的构造器。
如果说XML配置不符合你的口味,Spring还提供了基于Java的配置方法。如下所示。
package com.gaussic.knights; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; // @Configuration注解,声明这是一个配置类 @Configuration public class KnightConfig { // 与XML中的bean目的相同,声明一个SlayDragonQuest bean,传入System.out @Bean public Quest quest() { return new SlayDragonQuest(System.out); } // 与XML中的bean目的相同,声明一个BraveKnight bean,传入quest @Bean public Knight knight() { return new BraveKnight(quest()); } }
无论是使用基于XML还是基于Java的配置,DI的益处都是相同的。虽然BraveKnight依赖于Quest,但它并不知道将被给予什么样的Quest,或者这个Quest是从哪里来的。同样的,SlayDragonQuest依赖于PrintStream,但是它并不需要了解那个PrintStream是什么。只有Spring,通过它的配置,知道所有的组件是如何组织到一起的。这样,我们就可以在不修改当前类的情况下,去修改它的依赖。
这个例子展示了Spring中装配bean的一个简单方法。先不要在意太多的细节,我们将会在第二章深入探讨Spring配置。我们也将探讨Spring中装配bean的一些其他办法,包括一种让Spring自动发现bean和创建它们之间关系的方法。
现在你已经生命了BraveKnight和Quest之间的关系,你需要载入XML配置文件并且打开应用。
在Spring应用中,一个应用上下文(application context)载入bean的定义并将它们装配在一起。Spring应用上下文完全负责创建并装配所有构成应用的对象。Spring提供了多种应用上下文的实现,每一个主要的不同在于它们如何载入配置。
当bean被声明在xml文件中时,一个合适的方法是ClassPathXmlApplicationContext。这个Spring上下文实现从一个或多个(在应用的classpath目录的)XML文件载入Spring上下文。下面的main()方法使用了ClassPathXmlApplicationContext来载入knight.xml,并且获得对Knight对象的一个引用。
package com.gaussic.knights; import org.springframework.context.support.ClassPathXmlApplicationContext; public class KnightMain { public static void main(String[] args) throws Exception { // 从应用的classpath目录载入Spring配置文件 ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("META-INF/spring/knight.xml"); // 获取bean,并执行 Knight knight = context.getBean(Knight.class); knight.embarkOnQuest(); context.close(); } }
此处,main()方法基于knight.xml文件创建了Spring上下文。然后它利用这个上下文作为一个工厂来检索ID为knight的bean。利用对Knight对象的一个引用,它调用了embarkOnQuest()方法来使knight执行给予他的任务。注意这个类并不知道骑士被赋予了什么样的任务。这样,它也不知道这个任务是交给BraveKnight做的。只有knight.xml知道具体的实现是什么。
运行KnightMain的main()方法,将得到下列结果:
利用这个例子,你应该对依赖注入有了一定的了解。在这本书中你将看到更多的DI。现在让我们来看看Spring的另一个Java简化策略:基于方面(aspects)的声明式编程。
虽然DI可以松散的绑定软件的各个组件,面向方面编程(AOP)还可以让你更够捕捉那些在整个应用中可重用的组件。
AOP经常被定义为一项提升软件系统分离性(separation of concerns)的技术。系统往往由多个组件构成,每一个组件负责一个特定的功能。但是通常这些组件还需要承担其核心功能以外的附加功能。一些系统服务,例如日志、事务管理和安全,往往会进入一些其他的组件中,而这些组件往往负责其他的核心功能。这些系统服务通常被称为交叉相关(cross-cutting concerns),因为它们在系统中往往交叉在多个组件内。
由于将这些相关性传播到了多个组件中,你向你的代码引入了两种层次的复杂度:
实现系统级相关性的代码在多个组件中存在重复。这表明如果你需要改变这些相关性的工作方式,你需要访问多个组件。即使你将这个相关性抽象为一个独立的模块,这样其他的组件可以通过模块调用来使用它,这个方法调用依然在多个组件里面重复。
你的组件被与核心功能无关的代码污染了。一个向地址簿添加一个地址的方法,应该仅仅关心如何去添加这个地址,而不是还得去关心它的安全性和事务性。
下图展示了这一复杂度。左边的业务对象太过亲密地与右边的系统服务相联系。每个对象不仅仅知道自己被日志记录、安全检查,以及涉及到事务中,而且还需要负责自己去执行这些服务。
AOP可以模块化这些服务,然后声明式地将它们运用到需要它们的组件中。这样可以使得其他组件更加紧密地专注于负责它们自己的功能,完全无视任何系统服务的存在。简单的说,方面(aspects)使得POJOs保持普通(plain)。
把方面想象成一个覆盖多个组件和应用的毯子可以帮助理解,如下图所示。在其核心部分,一个包含多个模块的应用实现了业务功能。利用AOP,你可以利用一层层的其他功能将你的核心应用覆盖。这些层可以声明式地运用到你的应用中,以一种更为灵活的方式,而不需要让你的核心应用知道他们的存在。这是一个很强大的概念,因为它保证安全、事务管理和日志不去污染你的应用的核心业务逻辑。
为了说明方面在Spring中是如何运用的,让我们再来看看骑士的例子,向其中添加一个基本的Spring方面。
勇敢的骑士在屠杀巨龙和拯救少女归来之后,我们这些百姓要如何才能知道他的丰功伟绩呢?吟游诗人(minstrel)会将勇士的故事编成歌谣在民间传颂(有网络红人就必然要有网络推手啊)。我们假设你需要通过吟游诗人的服务来记录骑士出征和归来的整个过程。下面展示了这个Minstrel类:
package com.gaussic.knights; import java.io.PrintStream; // 吟游诗人 public class Minstrel { private PrintStream stream; public Minstrel(PrintStream stream) { this.stream = stream; } // 骑士出发前,颂唱 public void singBeforeQuest() { stream.println("Fa la la, the knight is so brave!"); } // 骑士归来,颂场 public void singAfterQuest() { stream.println("Tee hee hee, the brave knight did embark on a quest!"); } }
如你所见,Minstrel是一个有两个方法的简单类。singBeforeQuest()旨在骑士出发前调用,singAfrterQuest()旨在骑士完成任务归来调用。在这两种情况下,Minstrel都通过注入其构造器的PrintStrem来颂唱。
把这个Minstrel运用到你的代码中其实非常简单----你可以把它直接注入到BraveKnight中,对吗?让我们对BraveKnight做一点适当的调整,以使用Minstrel。下面的代码展示了将BraveKnight和Minstrel运用到一起的第一次尝试:
package com.gaussic.knights; public class BraveKnight2 { private Quest quest; private Minstrel minstrel; public BraveKnight2(Quest quest, Minstrel minstrel) { this.quest = quest; this.minstrel = minstrel; } // 一个骑士应该管理自己的吟游诗人吗??? public void embarkOnQuest() throws Exception { minstrel.singBeforeQuest(); quest.embark(); minstrel.singAfterQuest(); } }
上面的代码可以完成任务。现在你只需要回到Spring配置文件中去定义Minstrel bean,并将其注入到BraveKnight bean的构造器中。但是等等。。。。
好像有什么不对劲。。。一个骑士难道应该去管理他自己的吟游诗人吗?一个伟大的骑士是不会自己去传扬自身的功绩的,有节操的吟游诗人应该主动的去传颂伟大骑士的功绩,而不接受他人的收买。那么,为什么这里的骑士他时时地去提醒吟游诗人呢?
此外,由于这个骑士需要知道这个吟游诗人,你被强制将Minstrel注入到BraveKnight中。这不仅仅复杂化了BraveKnight的代码,还让我想到,如果你需要一个没有吟游诗人的骑士要怎么办?如果Minstrel是null那怎么办(骑士所处的地区,压根没有吟游诗人的存在)?那你是不是还要引入一些非空检查逻辑到你的代码里面呢?
你的简单的BraveKnight类开始变得越来越复杂(而骑士只想做一个纯粹的骑士)。但是,使用AOP,你可以声明,吟游诗人应该主动地去颂唱骑士的功绩,而骑士只需要做好本职工作。
为了把Minstrel转变为一个方面,你要做的仅仅是在Spring配置文件中声明它。这是一个更新版的knight.xml,加入了Minstrel方面的声明:
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> <!-- BraveKnight bean,id为knight --> <bean id="knight" class="com.gaussic.knights.BraveKnight"> <!-- 构造参数,引用 id为quest的bean --> <constructor-arg ref="quest"/> </bean> <!-- SlayDragonQuest bean,id为quest --> <bean id="quest" class="com.gaussic.knights.SlayDragonQuest"> <!-- 构造参数,值为 #{T(System).out} --> <constructor-arg value="#{T(System).out}"/> </bean> <!-- Minstrel bean, id为minstrel --> <bean id="minstrel" class="com.gaussic.knights.Minstrel"> <!-- 构造参数,值为 #{T(System).out} --> <constructor-arg value="#{T(System).out}" /> </bean> <!-- 方面配置 --> <aop:config> <aop:aspect ref="minstrel"> <!-- 在执行任何类的 embarkOnQuest()方法时调用 --> <aop:pointcut id="embark" expression="execution(* *.embarkOnQuest(..))"/> <!-- 在embark之前,调用 singBeforeQuest --> <aop:before pointcut-ref="embark" method="singBeforeQuest"/> <!-- 在embark之后,调用 singAfterQuest --> <aop:after pointcut-ref="embark" method="singAfterQuest"/> </aop:aspect> </aop:config> </beans>
这里你使用了Spring的 aop configuration 命名空间来声明Minstrel bean是一个方面。首先你声明Minstrel是一个bean。然后在<aop:aspect>元素引入那个bean。更进一步的定义这个aspect,你声明(使用 <aop:before>)了singBeforeQuest()方法在embarkOnQuest()方法前调用,称之为before advice。以及singAfterQuest()方法在embarkOnQuest()后调用,称之为 after advice。
在以上两种情况中,pointcut-ref属性都引用了一个名为embark的pointcut(切入点)。这个pointcut定义在了之前的<pointcut>元素中,使用了表达式属性级来选择这个advice应该作用在那个位置。表达式语法是AspectJ的pointcut表达式语言。
如果你不了解AspectJ,或者AspectJ pointcut表达式是如何编写的,不用担心。我们将在第4章对Spring AOP做更多的讲解。现在,知道你要请求Spring在BraveKnight在出征前后,调用Minstrel的singBeforeQuest()和singAfterQuest()方法足矣。
现在,不改动KnightMain的任何地方,运行main()方法,结果如下:
这就是它的全部了!只需要一点点的XML,你就将Minstrel转换为了一个Spring方面。如果它现在还没完全说明白,不用担心,你将在第4章中看到更多的Spring AOP示例,它们将给你更明确的概念。现在,这个例子有两个重点需要强调。
第一,Minstrel仍然是一个POJO,没有任何代码表明它将被作为一个方面。Minstrel只在你在Spring context中声明时,才是一个方面。
第二,也是最重要的,Minstrel可以运用到BraveKnight中,而BraveKnight不需要明确地调用它。实际上,BraveKnight甚至一直不知道还有Minstrel的存在。
我还应该指出,虽然你使用了一些Spring的魔法来将Minstrel转化为一个方面,你还是需要将它先声明为一个bean。重点是,你可以用Spring aspects做任何的事情,像其他的Spring beans一样,例如将依赖注入给它们。
使用aspects来颂唱骑士的功绩是很好玩的。但是Spring的AOP可以用在更多实用的地方。在后面你将看到,Spring AOP可以用来提供服务,例如声明式事务、安全,将在第9章和14章介绍。
在这之前,我们来看看Spring简化Java开发的另一个方法。
你是否写过一些看似之前写过的代码?这不是“似曾相似“,朋友。那是重复代码(boilerplate code)---为了执行常见或简单的任务需要一遍一遍的写相似的代码。
不幸的是,Java API的许多地方都含有重复的代码。使用JDBC做数据库查询就是一个明显的例子。如果你在使用JDBC,你可能需要写一些如下所示的代码:
public Employee getEmployeeById(long id) { Connection conn = null; PreparedStatement stmt = null; ResultSet rs = null; try { conn = dataSource.getConnection(); // 建立连接 // 选择 employee stmt = conn.prepareStatement("select id, firstname, lastname, salary from employee where id=?"); stmt.setLong(1, id); rs = stmt.executeQuery(); Employee employee = null; if (rs.next()) { employee = new Employee(); // 创建对象 employee.setId(rs.getLong("id")); employee.setFirstName(rs.getString("firstname")); employee.setLastName(rs.getString("lastname")); employee.setSalary(rs.getBigDecimal("salary")); } return employee; } catch (SQLException e) { // 错误处理 } finally { if (rs != null) { // 收拾残局 try { rs.close(); } catch (SQLException e) { } } if (stmt != null) { try { stmt.close(); } catch (SQLException e) { } } if (conn != null) { try { conn.close(); } catch (SQLException e) { } } } return null; }
如你所见,这个JDBC代码的目的是查询一个员工的名字和工资情况。由于这项特定的查询任务被埋藏在一堆的JDBC仪式中,你首先必须创建一个连接,然后创建一个陈述(statement),最后在查询结果。而且,为了安抚JDBC的脾气,你必须catch SQLException,其实就算它抛出了错误,你也没什么能够做的。
最后,在所有的事情都做完了,你还得收拾残局,关闭连接(connection)、陈述(statement)、结果集(result set)。这还是有可能出发JDBC的脾气,因而你还得catch SQLException。
上面代码最显著的问题是,如果你要做更多的JDBC查询,你将会写大量重复的代码。但是只有很少一部分是真正跟查询相关的,而JDBC重复代码要多得多。
在重复代码业务中,JDBC并不孤独。许多的工作都包含相似的重复代码。JMS、JNDI和大量的REST服务通常涉及到大量完全重复的代码。
Spring通过将重复代码封装在模板中来消除它们。Spring的JdbcTemplate可以在不需要所有传统JDBC仪式的情况下执行数据库操作。
例如,使用Spring的SimpleJdbcTemplate(JdbcTemplate的一个特定形式,利用了Java 5的优点),getEmployeeById()方法可以重写,使得它仅关注于检索员工数据的任务,而不需要考虑JDBC API的需求。下面的代码展示了更新后的getEmployeeById()的样子:
public Employee getEmployeeById(long id) { return jdbcTemplate.queryForObject( "select id, firstname, lastname, salary from employee where id=?", // SQL查询 new RowMapper<Employee>() { // 映射结果到对象 public Employee mapRow(ResultSet rs, int rowNum) throws SQLException { Employee employee = new Employee(); employee.setId(rs.getLong("id")); employee.setFirstName(rs.getString("firstname")); employee.setLastName(rs.getString("lastname")); employee.setSalary(rs.getBigDecimal("salary")); return employee; } }, id); }
如你所见,新版本的getEmployeeById()更加简洁且专注于从数据库中查询员工。模板的queryForObject()方法被给予一个SQL查询,一个RowMapper(为了将结果集数据映射到领域对象),以及0个或多个查询参数。你无法在getEmployeeById()中找到任何的JDBC重复代码。这一切都交给了模板来处理。
我已经想你展示了Spring是如何使用面向POJO的开发来降低Java开发复杂性的,DI, aspects 和 templates。同时,还展示了如何在XML配置文件中配置bean和aspect。但是如何载入这些文件呢?让我们看看Spring容器(container),存放应用的bean的地方。
在Spring应用中,你的应用对象住在Spring容器中。如图1.4所示,容器创建对象,然后将它们装配在一起,配置它们,然后管理它们的生命周期,从襁褓到坟墓。
在下一章节,你将了解如何配置Spring,使得它知道它需要创建、配置和装配什么对象。首先,知道对象什么时候hang out(闲逛?不好翻译)是很重要的。了解容器将帮助你了解对象是如何被管理的。
容器是Spring框架的核心。Spring的容器使用DI来管理组成一个应用的组件。这包括建立协调组件之间的联系。本身,这些组件更简洁切更易理解,它们支持重用,且易于单元测试。
没有单独的Spring容器。Spring包括多种的容器实现方法,可以分为两种独立的类型。Bean工厂(bean factory,由org.springframework.beans.factory.BeanFactory接口定义)是最简单的容器,提供DI的基本支持。应用上下文(application context,由org.springframework.context.ApplicationContext接口定义)建立与bean工厂的概念之上,提供应用框架服务,例如从配置文件中读取文本消息的能力和向感兴趣的事件监听器发送应用时间的能力。
虽然使用bean factory和application都可以,bean factory对于大部分应用来说还是太低级了。因此,相比bean factory,application context更受青睐。我们将专注于使用application context,而不花更多的时间在bean factory上。
Spring包含多种口味的应用上下文。以下是一部分你最有可能遇上的:
AnnotationConfigApplicationContext --- 从一个Java配置类中载入Spring应用上下文
AnnotationConfigWebApplicationContext --- 从一个Java配置类中载入Spring web应用上下文
ClassPathXmlApplicationContext --- 从一个或多个在classpath下的XML文件中载入Spring上下文,将上下文定义文件作为一个classpath资源文件
FileSystemXmlApplicationContext --- 从文件系统中的XML文件中载入上下文定义
XmlWebApplicationContext --- 从包含在一个web应用中的一个或多的XML文件中载入上下文定义
我们将在第八章讲解基于web的Spring应用时更加详细的说明AnnotationConfigWebApplicationContext和XmlWebApplicationContext。现在,让我们用FileSystemXmlApplicationContext从文件系统载入Spring上下文,或用ClassPathXmlApplicationContext从classpath载入Spring上下文。
从文件系统或classpath载入应用上下文与从bean factory中载入bean相类似。例如,下面是如何载入FileSystemXmlApplicationContext的:
ApplicationContext context = new FileSystemXmlApplicationContext("c:/knight.xml");
类似地,可以使用ClassPathXmlApplicationContext从应用的classpath载入应用上下文:
ApplicationContext context = new ClassPathXmlApplicationContext("knight.xml");
FileSystemXmlApplicationContext和ClassPathXmlApplicationContext的不同之处在于,前者在文件系统的特定的位置寻找knight.xml,而后者在应用的classpath中寻找knight.xml。
另外,如果你更偏向从Java配置中载入你的应用上下文的话,可以使用AnnotationConfigApplicationContext:
ApplicationContext context = new AnnotationConfigApplicationContext(KnightConfig.class);
取代指定一个载入Spring应用上下文的XML文件外,AnnotationConfigApplicationContext被给以一个配置类来载入bean。
现在手头上有了一个应用上下文,你可以利用上下文的getBean()方法,从Spring容器中检索bean了。
现在你知道了如何创建Spring容器的基本方法,让我们更进一步的看看在Bean容器中一个bean的生命周期。
在传统的Java应用中,bean的生命周期非常简单。Java的关键词被用来实例化bean,然后这个bean就可以使用了。一旦这个bean不再使用,将会被垃圾收集器处理。
相比之下,在Spring容器中的bean的生命周期更加复杂。理解Spring bean的生命周期是非常重要的,因为你可能需要利用Spring提供的一些有点来定制一个bean什么时候被创建。图1.5展示了一个典型的bean在载入到应用上下文时的生命周期。
如你所见,一个bean工厂在bean可以被使用前,执行了一系列的设置操作。让我们来探讨一些细节:
Spring实例化bean。
Spring将值和bean引用注入到bean的属性中。
如果一个bean实现了BeanNameAware,Spring将这个bean的id传递给setBeanName()方法。
如果一个bean实习了BeanFactoryAware,Spring调用setBeanFactory()方法,将一个引用传入一个附入(enclosing)的应用上下文。
如果一个bean实现了BeanPostProcessor接口,Spring调用其postProcessBeforeInitialization()方法。
如果一个bean实现了InitializingBean接口,Spring调用其afterPropertiesSet()方法。类似的,如果这个bean使用了init-method进行声明,那么特定的初始化方法将被调用。
如果一个bean实现了BeanPostProcessor,Spring调用其postProcessAfterInitialization()方法。
在这点,bean就可以被应用使用了,并且一直保持在应用上下文中,直到应用上下文被销毁。
如果一个bean实现了DisposableBean接口,Spring调用其destroy()方法。同样,如果这个bean被声明了destroy-method,将会调用特定的方法。
现在你已经知道了如何创建并载入一个Spring容器。但是一个空的容器本身并没有什么好处,它不含任何东西,除非你将什么放进去。为了实现Spring DI的好处,你必须将Spring容器中的应用对象装配起来。我们将在第二章节更加详细的讲解bean的装配。
首先,我们来调查一下现代Spring的蓝图(landscape),来看看Spring框架是由什么构成的,以及后面版本的Spring提供了什么新的特性。
如你所见,Spring框架专注于通过DI、AOP和减少重复来简化企业Java开发。即使那是Spring的全部,那也是值得一用的。但是对于Spring来说,还有很多令你难以置信的东西。
在Spring框架中,你将发现许多Spring简化Java开发的方法。但是,在Spring框架之上,是一个更大的建立在核心框架之上的项目生态系统,将Spring扩展到更多的领域,例如web services, REST, mobile和NoSQL。
让我们先来分解一下核心Spring框架,来看看它给我们带来了什么。然后我们再扩展我们的视野来看看Spring包的更多成员。
当你下载了Spring distriution之后,查看它的libs目录,你将发现多个JAR文件。在Spring 4.0中,有20个独立的模块,每个模块有3个JAR文件(binary class, source, JavaDoc)。完整的library JAR文件如图1.6所示。
这些模块可以被排成6个功能的类,如图1.7所示。
总的来说,这些模块给了你任何你需要的东西,来开发企业级应用。但是你不需要让你的应用完全基于Spring框架。你可以自由地选择适合你的应用的模块,并且在Spring无法满足你的需求时,选择其他的模块。Spring甚至提供了与许多其他框架和库的集成点,因而你不用去自己写它们。
让我们一个一个的来看看Spring的每一个模块,来了解一下每一个模块是如何拼凑整个Spring的版图的。
Spring框架的中心是一个容器,它负责管理Spring应用中的bean是如何创建、配置与管理的。这个模块的内部是Spring bean工厂,是Spring提供DI的一部分。建立在bean工厂之上,你将发现Spring应用上下文的多个实现,每一个都提供了配置Spring的不同方式。
除了bean工厂和应用上下文之外,这个模块还提供了许多企业级服务,例如邮件、JNDI访问、EJB集成,和调度。
Spring的所有模块都都是建立在核心容器之上的。你将在配置你的应用是隐式的使用这些类。我们将通篇讨论核心模块,从第二章开始,我们将更深入的挖掘Spring DI。
Spring在AOP模块中提供了面向方面编程的丰富支持。这个模块的作用服务于------为你自己的Spring应用开发你自己的aspects。与DI类似,AOP支持应用对象的松散耦合。但是利用AOP,应用级的相关性(例如事务和安全)解除了它们与其他对象的耦合。
我们将在第4章更加深入的讲解Spring AOP支持。