# 【第六章】 AOP 之 6.3 基于Schema的AOP ——跟我学spring3
## 6.3 基于Schema的AOP
基于Schema的AOP从Spring2.0之后通过“aop”命名空间来定义切面、切入点及声明通知。
在Spring配置文件中,所以AOP相关定义必须放在<aop:config>标签下,该标签下可以有<aop:pointcut>、<aop:advisor>、<aop:aspect>标签,配置顺序不可变。
* <aop:pointcut>:用来定义切入点,该切入点可以重用;
* <aop:advisor>:用来定义只有一个通知和一个切入点的切面;
* <aop:aspect>:用来定义切面,该切面可以包含多个切入点和通知,而且标签内部的通知和切入点定义是无序的;和advisor的区别就在此,advisor只包含一个通知和一个切入点。
![](https://box.kancloud.cn/2016-05-13_5735471e31be8.JPG)
6.3.1 声明切面
切面就是包含切入点和通知的对象,在Spring容器中将被定义为一个Bean,Schema方式的切面需要一个切面支持Bean,该支持Bean的字段和方法提供了切面的状态和行为信息,并通过配置方式来指定切入点和通知实现。
切面使用<aop:aspect>标签指定,ref属性用来引用切面支持Bean。
切面支持Bean“aspectSupportBean”跟普通Bean完全一样使用,切面使用“ref”属性引用它。
### 6.3.2 声明切入点
切入点在Spring中也是一个Bean,Bean定义方式可以有很三种方式:
**1)在<aop:config>标签下使用<aop:pointcut>声明一个切入点Bean**,该切入点可以被多个切面使用,对于需要共享使用的切入点最好使用该方式,该切入点使用id属性指定Bean名字,在通知定义时使用pointcut-ref属性通过该id引用切入点,expression属性指定切入点表达式:
1. <aop:config>
2. <aop:pointcut id="pointcut" expression="execution(* cn.javass..*.*(..))"/>
3. <aop:aspect ref="aspectSupportBean">
4. <aop:before pointcut-ref="pointcut" method="before"/>
5. </aop:aspect>
6. </aop:config>
**2)在<aop:aspect>标签下使用<aop:pointcut>声明一个切入点Bean**,该切入点可以被多个切面使用,但一般该切入点只被该切面使用,当然也可以被其他切面使用,但最好不要那样使用,该切入点使用id属性指定Bean名字,在通知定义时使用pointcut-ref属性通过该id引用切入点,expression属性指定切入点表达式:
1. <aop:config>
2. <aop:aspect ref="aspectSupportBean">
3. <aop:pointcut id=" pointcut" expression="execution(* cn.javass..*.*(..))"/>
4. <aop:before pointcut-ref="pointcut" method="before"/>
5. </aop:aspect>
6. </aop:config>
**3)匿名切入点Bean,**可以在声明通知时通过pointcut属性指定切入点表达式,该切入点是匿名切入点,只被该通知使用:
1. <aop:config>
2. <aop:aspect ref="aspectSupportBean">
3. <aop:after pointcut="execution(* cn.javass..*.*(..))" method="afterFinallyAdvice"/>
4. </aop:aspect>
5. </aop:config>
### 6.3.3 声明通知
基于Schema方式支持前边介绍的5中通知类型:
**一、前置通知:**在切入点选择的方法之前执行,通过<aop:aspect>标签下的<aop:before>标签声明:
1. <aop:before pointcut="切入点表达式" pointcut-ref="切入点Bean引用"
2. method="前置通知实现方法名"
3. arg-names="前置通知实现方法参数列表参数名字"/>
**pointcut和pointcut-ref:**二者选一,指定切入点;
**method:**指定前置通知实现方法名,如果是多态需要加上参数类型,多个用“,”隔开,如beforeAdvice(java.lang.String);
**arg-names:**指定通知实现方法的参数名字,多个用“,”分隔,可选,类似于【3.1.2 构造器注入】中的参数名注入限制:**在class文件中没生成变量调试信息是获取不到方法参数名字的,因此只有在类没生成变量调试信息时才需要使用arg-names属性来指定参数名,如**arg-names="param"表示通知实现方法的参数列表的第一个参数名字为“param”。
首先在cn.javass.spring.chapter6.service.IhelloWorldService定义一个测试方法:
1. public void sayBefore(String param);
其次在cn.javass.spring.chapter6.service.impl. HelloWorldService定义实现
1. @Override
2. public void sayBefore(String param) {
3. System.out.println("============say " + param);
4. }
第三在cn.javass.spring.chapter6.aop. HelloWorldAspect定义通知实现:
1. public void beforeAdvice(String param) {
2. System.out.println("===========before advice param:" + param);
3. }
最后在chapter6/advice.xml配置文件中进行如下配置:
1. <bean id="helloWorldService" class="cn.javass.spring.chapter6.service.impl.HelloWorldService"/>
2. <bean id="aspect" class="cn.javass.spring.chapter6.aop.HelloWorldAspect"/>
3. <aop:config>
4. <aop:aspect ref="aspect">
5. <aop:before pointcut="execution(* cn.javass..*.sayBefore(..)) and args(param)"
6. method="beforeAdvice(java.lang.String)"
7. arg-names="param"/>
8. </aop:aspect>
9. </aop:config>
测试代码cn.javass.spring.chapter6.AopTest:
1. @Test
2. public void testSchemaBeforeAdvice(){
3. System.out.println("======================================");
4. ApplicationContext ctx = new ClassPathXmlApplicationContext("chapter6/advice.xml");
5. IHelloWorldService helloworldService = ctx.getBean("helloWorldService", IHelloWorldService.class);
6. helloworldService.sayBefore("before");
7. System.out.println("======================================");
8. }
将输入:
```
==========================================
===========before advice param:before
============say before
==========================================
```
分析一下吧:
**1)切入点匹配:**在配置中使用“execution(* cn.javass..*.sayBefore(..)) ”匹配目标方法sayBefore,且使用“args(param)”匹配目标方法只有一个参数且传入的参数类型为通知实现方法中同名的参数类型;
**2)目标方法定义:**使用method=" beforeAdvice(java.lang.String) "指定前置通知实现方法,且该通知有一个参数类型为java.lang.String参数;
**3)目标方法参数命名:**其中使用arg-names=" param "指定通知实现方法参数名为“param”,切入点中使用“args(param)”匹配的目标方法参数将自动传递给通知实现方法同名参数。
**二、后置返回通知:**在切入点选择的方法正常返回时执行,通过<aop:aspect>标签下的<aop:after-returning>标签声明:
1. <aop:after-returning pointcut="切入点表达式" pointcut-ref="切入点Bean引用"
2. method="后置返回通知实现方法名"
3. arg-names="后置返回通知实现方法参数列表参数名字"
4. returning="返回值对应的后置返回通知实现方法参数名"
5. />
**pointcut和pointcut-ref:**同前置通知同义;
**method:**同前置通知同义;
**arg-names:**同前置通知同义;
**returning:**定义一个名字,该名字用于匹配通知实现方法的一个参数名,当目标方法执行正常返回后,将把目标方法返回值传给通知方法;returning限定了只有目标方法返回值匹配与通知方法相应参数类型时才能执行后置返回通知,否则不执行,对于returning对应的通知方法参数为Object类型将匹配任何目标返回值。
首先在cn.javass.spring.chapter6.service.IhelloWorldService定义一个测试方法:
1. public boolean sayAfterReturning();
其次在cn.javass.spring.chapter6.service.impl. HelloWorldService定义实现
1. @Override
2. public boolean sayAfterReturning() {
3. System.out.println("============after returning");
4. return true;
5. }
第三在cn.javass.spring.chapter6.aop. HelloWorldAspect定义通知实现:
1. public void afterReturningAdvice(Object retVal) {
2. System.out.println("===========after returning advice retVal:" + retVal);
3. }
最后在chapter6/advice.xml配置文件中接着前置通知配置的例子添加如下配置:
1. <aop:after-returning pointcut="execution(* cn.javass..*.sayAfterReturning(..))"
2. method="afterReturningAdvice"
3. arg-names="retVal"
4. returning="retVal"/>
测试代码cn.javass.spring.chapter6.AopTest:
1. @Test
2. public void testSchemaAfterReturningAdvice() {
3. System.out.println("======================================");
4. ApplicationContext ctx = new ClassPathXmlApplicationContext("chapter6/advice.xml");
5. IHelloWorldService helloworldService = ctx.getBean("helloWorldService", IHelloWorldService.class);
6. helloworldService.sayAfterReturning();
7. System.out.println("======================================");
8. }
将输入:
```
======================================
============after returning
===========after returning advice retVal:true
======================================
```
分析一下吧:
**1)切入点匹配:**在配置中使用“execution(* cn.javass..*.sayAfterReturning(..)) ”匹配目标方法sayAfterReturning,该方法返回true;
**2)目标方法定义:**使用method="afterReturningAdvice"指定后置返回通知实现方法;
**3)目标方法参数命名:**其中使用arg-names="retVal"指定通知实现方法参数名为“retVal”;
**4)返回值命名:**returning="retVal"用于将目标返回值赋值给通知实现方法参数名为“retVal”的参数上。
**三、后置异常通知:**在切入点选择的方法抛出异常时执行,通过<aop:aspect>标签下的<aop:after-throwing>标签声明:
1. <aop:after-throwing pointcut="切入点表达式" pointcut-ref="切入点Bean引用"
2. method="后置异常通知实现方法名"
3. arg-names="后置异常通知实现方法参数列表参数名字"
4. throwing="将抛出的异常赋值给的通知实现方法参数名"/>
**pointcut和pointcut-ref:**同前置通知同义;
**method:**同前置通知同义;
**arg-names:**同前置通知同义;
**throwing:**定义一个名字,该名字用于匹配通知实现方法的一个参数名,当目标方法抛出异常返回后,将把目标方法抛出的异常传给通知方法;throwing限定了只有目标方法抛出的异常匹配与通知方法相应参数异常类型时才能执行后置异常通知,否则不执行,对于throwing对应的通知方法参数为Throwable类型将匹配任何异常。
首先在cn.javass.spring.chapter6.service.IhelloWorldService定义一个测试方法:
1. public void sayAfterThrowing();
其次在cn.javass.spring.chapter6.service.impl. HelloWorldService定义实现
1. @Override
2. public void sayAfterThrowing() {
3. System.out.println("============before throwing");
4. throw new RuntimeException();
5. }
第三在cn.javass.spring.chapter6.aop. HelloWorldAspect定义通知实现:
1. public void afterThrowingAdvice(Exception exception) {
2. System.out.println("===========after throwing advice exception:" + exception);
3. }
最后在chapter6/advice.xml配置文件中接着前置通知配置的例子添加如下配置:
1. <aop:after-throwing pointcut="execution(* cn.javass..*.sayAfterThrowing(..))"
2. method="afterThrowingAdvice"
3. arg-names="exception"
4. throwing="exception"/>
测试代码cn.javass.spring.chapter6.AopTest:
1. @Test(expected = RuntimeException.class)
2. public void testSchemaAfterThrowingAdvice() {
3. System.out.println("======================================");
4. ApplicationContext ctx = new ClassPathXmlApplicationContext("chapter6/advice.xml");
5. IHelloWorldService helloworldService = ctx.getBean("helloWorldService", IHelloWorldService.class);
6. helloworldService.sayAfterThrowing();
7. System.out.println("======================================");
8. }
将输入:
```
======================================
============before throwing
===========after throwing advice exception:java.lang.RuntimeException
======================================
```
分析一下吧:
**1)切入点匹配:**在配置中使用“execution(* cn.javass..*.sayAfterThrowing(..))”匹配目标方法sayAfterThrowing,该方法将抛出RuntimeException异常;
**2)目标方法定义:**使用method="afterThrowingAdvice"指定后置异常通知实现方法;
**3)目标方法参数命名:**其中使用arg-names="exception"指定通知实现方法参数名为“exception”;
**4)异常命名:**returning="exception"用于将目标方法抛出的异常赋值给通知实现方法参数名为“exception”的参数上。
**四、后置最终通知:**在切入点选择的方法返回时执行,不管是正常返回还是抛出异常都执行,通过<aop:aspect>标签下的<aop:after >标签声明:
1. <aop:after pointcut="切入点表达式" pointcut-ref="切入点Bean引用"
2. method="后置最终通知实现方法名"
3. arg-names="后置最终通知实现方法参数列表参数名字"/>
**pointcut和pointcut-ref:**同前置通知同义;
**method:**同前置通知同义;
**arg-names:**同前置通知同义;
首先在cn.javass.spring.chapter6.service.IhelloWorldService定义一个测试方法:
1. public boolean sayAfterFinally();
其次在cn.javass.spring.chapter6.service.impl. HelloWorldService定义实现
1. @Override
2. public boolean sayAfterFinally() {
3. System.out.println("============before finally");
4. throw new RuntimeException();
5. }
第三在cn.javass.spring.chapter6.aop. HelloWorldAspect定义通知实现:
1. public void afterFinallyAdvice() {
2. System.out.println("===========after finally advice");
3. }
最后在chapter6/advice.xml配置文件中接着前置通知配置的例子添加如下配置:
1. <aop:after pointcut="execution(* cn.javass..*.sayAfterFinally(..))"
2. method="afterFinallyAdvice"/>
测试代码cn.javass.spring.chapter6.AopTest:
1. @Test(expected = RuntimeException.class)
2. public void testSchemaAfterFinallyAdvice() {
3. System.out.println("======================================");
4. ApplicationContext ctx = new ClassPathXmlApplicationContext("chapter6/advice.xml");
5. IHelloWorldService helloworldService = ctx.getBean("helloWorldService", IHelloWorldService.class);
6. helloworldService.sayAfterFinally();
7. System.out.println("======================================");
8. }
将输入:
```
======================================
============before finally
===========after finally advice
======================================
```
分析一下吧:
**1)切入点匹配:**在配置中使用“execution(* cn.javass..*.sayAfterFinally(..))”匹配目标方法sayAfterFinally,该方法将抛出RuntimeException异常;
**2)目标方法定义:**使用method=" afterFinallyAdvice "指定后置最终通知实现方法。
**五、环绕通知:**环绕着在切入点选择的连接点处的方法所执行的通知,环绕通知非常强大,可以决定目标方法是否执行,什么时候执行,执行时是否需要替换方法参数,执行完毕是否需要替换返回值,可通过<aop:aspect>标签下的<aop:around >标签声明:
1. <aop:around pointcut="切入点表达式" pointcut-ref="切入点Bean引用"
2. method="后置最终通知实现方法名"
3. arg-names="后置最终通知实现方法参数列表参数名字"/>
**pointcut和pointcut-ref:**同前置通知同义;
**method:**同前置通知同义;
**arg-names:**同前置通知同义;
环绕通知第一个参数必须是org.aspectj.lang.ProceedingJoinPoint类型,在通知实现方法内部使用ProceedingJoinPoint的proceed()方法使目标方法执行,proceed 方法可以传入可选的Object[]数组,该数组的值将被作为目标方法执行时的参数。
首先在cn.javass.spring.chapter6.service.IhelloWorldService定义一个测试方法:
1. public void sayAround(String param);
其次在cn.javass.spring.chapter6.service.impl. HelloWorldService定义实现
1. @Override
2. public void sayAround(String param) {
3. System.out.println("============around param:" + param);
4. }
第三在cn.javass.spring.chapter6.aop. HelloWorldAspect定义通知实现:
1. public Object aroundAdvice(ProceedingJoinPoint pjp) throws Throwable {
2. System.out.println("===========around before advice");
3. Object retVal = pjp.proceed(new Object[] {"replace"});
4. System.out.println("===========around after advice");
5. return retVal;
6. }
最后在chapter6/advice.xml配置文件中接着前置通知配置的例子添加如下配置:
1. <aop:around pointcut="execution(* cn.javass..*.sayAround(..))"
2. method="aroundAdvice"/>
测试代码cn.javass.spring.chapter6.AopTest:
1. @Test
2. public void testSchemaAroundAdvice() {
3. System.out.println("======================================");
4. ApplicationContext ctx = new ClassPathXmlApplicationContext("chapter6/advice.xml");
5. IHelloWorldService helloworldService =
6. ctx.getBean("helloWorldService", IHelloWorldService.class);
7. helloworldService.sayAround("haha");
8. System.out.println("======================================");
9. }
将输入:
```
======================================
===========around before advice
============around param:replace
===========around after advice
======================================
```
分析一下吧:
**1)切入点匹配:**在配置中使用“execution(* cn.javass..*.sayAround(..))”匹配目标方法sayAround;
**2)目标方法定义:**使用method="aroundAdvice"指定环绕通知实现方法,在该实现中,第一个方法参数为pjp,类型为ProceedingJoinPoint,其中“Object retVal = pjp.proceed(new Object[] {"replace"});”,用于执行目标方法,且目标方法参数被“new Object[] {"replace"}”替换,最后返回“retVal ”返回值。
**3)测试:**我们使用“helloworldService.sayAround("haha");”传入参数为“haha”,但最终输出为“replace”,说明参数被替换了。
### 6.3.4 引入
Spring引入允许为目标对象引入新的接口,通过在< aop:aspect>标签内使用< aop:declare-parents>标签进行引入,定义方式如下:
1. <aop:declare-parents
2. types-matching="AspectJ语法类型表达式"
3. implement-interface=引入的接口"
4. default-impl="引入接口的默认实现"
5. delegate-ref="引入接口的默认实现Bean引用"/>
**types-matching:**匹配需要引入接口的目标对象的AspectJ语法类型表达式;
**implement-interface:**定义需要引入的接口;
**default-impl和delegate-ref:**定义引入接口的默认实现,二者选一,default-impl是接口的默认实现类全限定名,而delegate-ref是默认的实现的委托Bean名;
接下来让我们练习一下吧:
首先定义引入的接口及默认实现:
1. package cn.javass.spring.chapter6.service;
2. public interface IIntroductionService {
3. public void induct();
4. }
1. package cn.javass.spring.chapter6.service.impl;
2. import cn.javass.spring.chapter6.service.IIntroductionService;
3. public class IntroductiondService implements IIntroductionService {
4. @Override
5. public void induct() {
6. System.out.println("=========introduction");
7. }
8. }
其次在chapter6/advice.xml配置文件中接着前置通知配置的例子添加如下配置:
1. <aop:declare-parents
2. types-matching="cn.javass..*.IHelloWorldService+"
3. implement-interface="cn.javass.spring.chapter6.service.IIntroductionService"
4. default-impl="cn.javass.spring.chapter6.service.impl.IntroductiondService"/>
最后测试一下吧,测试代码cn.javass.spring.chapter6.AopTest:
1. @Test
2. public void testSchemaIntroduction() {
3. System.out.println("======================================");
4. ApplicationContext ctx = new ClassPathXmlApplicationContext("chapter6/advice.xml");
5. IIntroductionService introductionService =
6. ctx.getBean("helloWorldService", IIntroductionService.class);
7. introductionService.induct();
8. System.out.println("======================================");
9. }
将输入:
```
======================================
=========introduction
======================================
```
分析一下吧:
**1)目标对象类型匹配:**使用types-matching="cn.javass..*.IHelloWorldService+"匹配IHelloWorldService接口的子类型,如HelloWorldService实现;
**2)引入接口定义:**通过implement-interface属性表示引入的接口,如“cn.javass.spring.chapter6.service.IIntroductionService”。
**3)引入接口的实现:**通过default-impl属性指定,如“cn.javass.spring.chapter6.service.impl.IntroductiondService”,也可以使用“delegate-ref”来指定实现的Bean。
**4)获取引入接口:**如使用“ctx.getBean("helloWorldService", IIntroductionService.class);”可直接获取到引入的接口。
### 6.3.5 Advisor
Advisor表示只有一个通知和一个切入点的切面,由于Spring AOP都是基于AOP联盟的拦截器模型的环绕通知的,所以引入Advisor来支持各种通知类型(如前置通知等5种),Advisor概念来自于Spring1.2对AOP的支持,在AspectJ中没有相应的概念对应。
Advisor可以使用<aop:config>标签下的<aop:advisor>标签定义:
1. <aop:advisor pointcut="切入点表达式" pointcut-ref="切入点Bean引用"
2. advice-ref="通知API实现引用"/>
**pointcut和pointcut-ref:**二者选一,指定切入点表达式;
**advice-ref:**引用通知API实现Bean,如前置通知接口为MethodBeforeAdvice;
接下来让我们看一下示例吧:
首先在cn.javass.spring.chapter6.service.IhelloWorldService定义一个测试方法:
1. public void sayAdvisorBefore(String param);
其次在cn.javass.spring.chapter6.service.impl. HelloWorldService定义实现
1. @Override
2. public void sayAdvisorBefore(String param) {
3. System.out.println("============say " + param);
4. }
第三定义前置通知API实现:
1. package cn.javass.spring.chapter6.aop;
2. import java.lang.reflect.Method;
3. import org.springframework.aop.MethodBeforeAdvice;
4. public class BeforeAdviceImpl implements MethodBeforeAdvice {
5. @Override
6. public void before(Method method, Object[] args, Object target) throws Throwable {
7. System.out.println("===========before advice");
8. }
9. }
在chapter6/advice.xml配置文件中先添加通知实现Bean定义:
2. <bean id="beforeAdvice" class="cn.javass.spring.chapter6.aop.BeforeAdviceImpl"/>
然后在<aop:config>标签下,添加Advisor定义,添加时注意顺序:
1. <aop:advisor pointcut="execution(* cn.javass..*.sayAdvisorBefore(..))"
2. advice-ref="beforeAdvice"/>
测试代码cn.javass.spring.chapter6.AopTest:
1. @Test
2. public void testSchemaAdvisor() {
3. System.out.println("======================================");
4. ApplicationContext ctx = new ClassPathXmlApplicationContext("chapter6/advice.xml");
5. IHelloWorldService helloworldService =
6. ctx.getBean("helloWorldService", IHelloWorldService.class);
7. helloworldService.sayAdvisorBefore("haha");
8. System.out.println("======================================");
9. }
将输入:
```
======================================
===========before advice
============say haha
======================================
```
在此我们只介绍了前置通知API,其他类型的在后边章节介绍。
不推荐使用Advisor,除了在进行事务控制的情况下,其他情况一般不推荐使用该方式,该方式属于侵入式设计,必须实现通知API。
- 跟我学 Spring3
- 【第二章】 IoC 之 2.1 IoC基础 ——跟我学Spring3
- 【第二章】 IoC 之 2.2 IoC 容器基本原理 ——跟我学Spring3
- 【第二章】 IoC 之 2.3 IoC的配置使用——跟我学Spring3
- 【第三章】 DI 之 3.1 DI的配置使用 ——跟我学spring3
- 【第三章】 DI 之 3.2 循环依赖 ——跟我学spring3
- 【第三章】 DI 之 3.3 更多DI的知识 ——跟我学spring3
- 【第三章】 DI 之 3.4 Bean的作用域 ——跟我学spring3
- 【第四章】 资源 之 4.1 基础知识 ——跟我学spring3
- 【第四章】 资源 之 4.2 内置Resource实现 ——跟我学spring3
- 【第四章】 资源 之 4.3 访问Resource ——跟我学spring3
- 【第四章】 资源 之 4.4 Resource通配符路径 ——跟我学spring3
- 【第五章】Spring表达式语言 之 5.1 概述 5.2 SpEL基础 ——跟我学spring3
- 【第五章】Spring表达式语言 之 5.3 SpEL语法 ——跟我学spring3
- 【第五章】Spring表达式语言 之 5.4在Bean定义中使用EL—跟我学spring3
- 【第六章】 AOP 之 6.1 AOP基础 ——跟我学spring3
- 【第六章】 AOP 之 6.2 AOP的HelloWorld ——跟我学spring3
- 【第六章】 AOP 之 6.3 基于Schema的AOP ——跟我学spring3
- 【第六章】 AOP 之 6.4 基于@AspectJ的AOP ——跟我学spring3
- 【第六章】 AOP 之 6.5 AspectJ切入点语法详解 ——跟我学spring3
- 【第六章】 AOP 之 6.6 通知参数 ——跟我学spring3
- 【第六章】 AOP 之 6.7 通知顺序 ——跟我学spring3
- 【第六章】 AOP 之 6.8 切面实例化模型 ——跟我学spring3
- 【第六章】 AOP 之 6.9 代理机制 ——跟我学spring3
- 【第七章】 对JDBC的支持 之 7.1 概述 ——跟我学spring3
- 【第七章】 对JDBC的支持 之 7.2 JDBC模板类 ——跟我学spring3
- 【第七章】 对JDBC的支持 之 7.3 关系数据库操作对象化 ——跟我学spring3
- 【第七章】 对JDBC的支持 之 7.4 Spring提供的其它帮助 ——跟我学spring3【私塾在线原创】
- 【第七章】 对JDBC的支持 之 7.5 集成Spring JDBC及最佳实践 ——跟我学spring3
- 【第八章】 对ORM的支持 之 8.1 概述 ——跟我学spring3
- 【第八章】 对ORM的支持 之 8.2 集成Hibernate3 ——跟我学spring3
- 【第八章】 对ORM的支持 之 8.3 集成iBATIS ——跟我学spring3
- 【第八章】 对ORM的支持 之 8.4 集成JPA ——跟我学spring3
- 【第九章】 Spring的事务 之 9.1 数据库事务概述 ——跟我学spring3
- 【第九章】 Spring的事务 之 9.2 事务管理器 ——跟我学spring3
- 【第九章】 Spring的事务 之 9.3 编程式事务 ——跟我学spring3
- 【第九章】 Spring的事务 之 9.4 声明式事务 ——跟我学spring3
- 【第十章】集成其它Web框架 之 10.1 概述 ——跟我学spring3
- 【第十章】集成其它Web框架 之 10.2 集成Struts1.x ——跟我学spring3
- 【第十章】集成其它Web框架 之 10.3 集成Struts2.x ——跟我学spring3
- 【第十章】集成其它Web框架 之 10.4 集成JSF ——跟我学spring3
- 【第十一章】 SSH集成开发积分商城 之 11.1 概述 ——跟我学spring3
- 【第十一章】 SSH集成开发积分商城 之 11.2 实现通用层 ——跟我学spring3
- 【第十一章】 SSH集成开发积分商城 之 11.3 实现积分商城层 ——跟我学spring3
- 【第十二章】零配置 之 12.1 概述 ——跟我学spring3
- 【第十二章】零配置 之 12.2 注解实现Bean依赖注入 ——跟我学spring3
- 【第十二章】零配置 之 12.3 注解实现Bean定义 ——跟我学spring3
- 【第十二章】零配置 之 12.4 基于Java类定义Bean配置元数据 ——跟我学spring3
- 【第十二章】零配置 之 12.5 综合示例-积分商城 ——跟我学spring3
- 【第十三章】 测试 之 13.1 概述 13.2 单元测试 ——跟我学spring3
- 【第十三章】 测试 之 13.3 集成测试 ——跟我学spring3
- 跟我学 Spring MVC
- SpringMVC + spring3.1.1 + hibernate4.1.0 集成及常见问题总结
- Spring Web MVC中的页面缓存支持 ——跟我学SpringMVC系列
- Spring3 Web MVC下的数据类型转换(第一篇)——《跟我学Spring3 Web MVC》抢先看
- Spring3 Web MVC下的数据格式化(第二篇)——《跟我学Spring3 Web MVC》抢先看
- 第一章 Web MVC简介 —— 跟开涛学SpringMVC
- 第二章 Spring MVC入门 —— 跟开涛学SpringMVC
- 第三章 DispatcherServlet详解 ——跟开涛学SpringMVC
- 第四章 Controller接口控制器详解(1)——跟着开涛学SpringMVC
- 第四章 Controller接口控制器详解(2)——跟着开涛学SpringMVC
- 第四章 Controller接口控制器详解(3)——跟着开涛学SpringMVC
- 第四章 Controller接口控制器详解 (4)——跟着开涛学SpringMVC
- 第四章 Controller接口控制器详解(5)——跟着开涛学SpringMVC
- 跟着开涛学SpringMVC 第一章源代码下载
- 第二章 Spring MVC入门 源代码下载
- 第四章 Controller接口控制器详解 源代码下载
- 第四章 Controller接口控制器详解(6)——跟着开涛学SpringMVC
- 第四章 Controller接口控制器详解(7 完)——跟着开涛学SpringMVC
- 第五章 处理器拦截器详解——跟着开涛学SpringMVC
- 源代码下载 第五章 处理器拦截器详解——跟着开涛学SpringMVC
- 注解式控制器运行流程及处理器定义 第六章 注解式控制器详解——跟着开涛学SpringMVC
- 源代码下载 第六章 注解式控制器详解
- SpringMVC3强大的请求映射规则详解 第六章 注解式控制器详解——跟着开涛学SpringMVC
- Spring MVC 3.1新特性 生产者、消费者请求限定 —— 第六章 注解式控制器详解——跟着开涛学SpringMVC
- SpringMVC强大的数据绑定(1)——第六章 注解式控制器详解——跟着开涛学SpringMVC
- SpringMVC强大的数据绑定(2)——第六章 注解式控制器详解——跟着开涛学SpringMVC
- SpringMVC数据类型转换——第七章 注解式控制器的数据验证、类型转换及格式化——跟着开涛学SpringMVC
- SpringMVC数据格式化——第七章 注解式控制器的数据验证、类型转换及格式化——跟着开涛学SpringMVC
- SpringMVC数据验证——第七章 注解式控制器的数据验证、类型转换及格式化——跟着开涛学SpringMVC