Spring---AOP应用

1、项目应用

1.1 定义目标类

1)定义目标接口:

public interface IHelloWorldService {  
    public void sayHello();  
}  


2)定义目标接口实现:

public class HelloWorldService implements IHelloWorldService {  
    @Override  
    public void sayHello() {  
        System.out.println("============Hello World!");  
    }  
}
注:在日常开发中最后将业务逻辑定义在一个专门的service包下,而实现定义在service包下的impl包中,服务接口以IXXXService形式,而服务实现就是XXXService,这就是规约设计,见名知义。当然可以使用公司内部更好的形式,只要大家都好理解就可以了。 


1.2 定义切面支持类

有了目标类,该定义切面了,切面就是通知和切入点的组合,而切面是通过配置方式定义的,因此这定义切面前,我们需要定义切面支持类,切面支持类提供了通知实现:

public class HelloWorldAspect {  

    //前置通知  
    public void beforeAdvice() {  
        System.out.println("===========before advice");  
    }  

    //后置最终通知  
    public void afterFinallyAdvice() {  
        System.out.println("===========after finally advice");  
    }  
}
此处HelloWorldAspect类不是真正的切面实现,只是定义了通知实现的类,在此我们可以把它看作就是缺少了切入点的切面
 
注:对于AOP相关类最好专门放到一个包下,如“aop”包,因为AOP是动态织入的,所以如果某个目标类被AOP拦截了并应用了通知,可能很难发现这个通知实现在哪个包里,因此推荐使用规约命名,方便以后维护人员查找相应的AOP实现。


1.3  在XML中进行配置切面

有了通知实现,那就让我们来配置切面吧:

<!--首先配置AOP需要aop命名空间,配置头如下-->
<?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-3.0.xsd  
           http://www.springframework.org/schema/aop  
           http://www.springframework.org/schema/aop/spring-aop-3.0.xsd">  

<!--配置目标类-->
<bean id="helloWorldService"  class="cn.javass.spring.chapter6.service.impl.HelloWorldService"/>

<!--配置切面-->
<bean id="aspect" class="cn.javass.spring.chapter6.aop.HelloWorldAspect"/>  
<aop:config>  
    <aop:pointcut id="pointcut" expression="execution(* cn.javass..*.*(..))"/>  
    <aop:aspect ref="aspect">  
        <aop:before pointcut-ref="pointcut" method="beforeAdvice"/>  
        <aop:after pointcut="execution(* cn.javass..*.*(..))" method="afterFinallyAdvice"/>  
    </aop:aspect>  
</aop:config>  

 
</beans>  


切入点(在哪加功能)使用<aop:config>标签下的<aop:pointcut>配置,

expression属性用于定义切入点模式,默认是AspectJ语法,“execution(* cn.javass..*.*(..))”表示匹配cn.javass包及子包下的任何方法执行。
 
切面(在哪加什么功能==切入点+通知)使用<aop:config>标签下的<aop:aspect>标签配置,其中“ref”用来引用切面支持类的方法。
 
前置通知使用<aop:aspect>标签下的<aop:before>标签来定义,pointcut-ref属性用于引用切入点Bean(在哪),而method用来引用切面通知实现类中的方法(加什么功能),该方法就是通知实现,即在目标类方法执行之前调用的方法。
 
最终通知使用<aop:aspect>标签下的<aop:after >标签来定义,切入点除了使用pointcut-ref属性来引用已经存在的切入点,也可以使用pointcut属性来定义,如pointcut="execution(* cn.javass..*.*(..))",method属性同样是指定通知实现,即在目标类方法执行之后调用的方法。


1.4 运行测试

测试类非常简单,调用被代理Bean跟调用普通Bean完全一样,Spring AOP将为目标对象创建AOP代理,具体测试代码如下:

import org.junit.Test;  
import org.springframework.context.ApplicationContext;  
import org.springframework.context.support.ClassPathXmlApplicationContext;  
import cn.javass.spring.chapter6.service.IHelloWorldService;  
import cn.javass.spring.chapter6.service.IPayService;  
public class AopTest {  
    @Test  
    public void testHelloworld() {  
        ApplicationContext ctx =  new ClassPathXmlApplicationContext("chapter6/helloworld.xml");  
        IHelloWorldService helloworldService =  
        ctx.getBean("helloWorldService", IHelloWorldService.class);  
        helloworldService.sayHello();  
    }  
}  
该测试将输出如下如下内容:

===========before advice  
============Hello World!  
===========after finally advice 

从输出我们可以看出:前置通知在切入点选择的连接点(方法)之前允许,而后置通知将在连接点(方法)之后执行


2、基于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只包含一个通知和一个切入点。


2.1 声明切面

切面就是包含切入点和通知的对象,在Spring容器中将被定义为一个Bean,Schema方式的切面需要一个切面支持Bean,该支持Bean的字段和方法提供了切面的状态和行为信息,并通过配置方式来指定切入点和通知实现。
 
切面使用<aop:aspect>标签指定,ref属性用来引用切面支持Bean。切面支持Bean“aspectSupportBean”跟普通Bean完全一样使用,切面使用“ref”属性引用它。

2.2 声明切入点
    切入点在Spring中也是一个Bean,Bean定义方式可以有很三种方式:
    1)在<aop:config>标签下使用<aop:pointcut>声明一个切入点Bean,该切入点可以被多个切面使用,对于需要共享使用的切入点最好使用该方式,该切入点使用id属性指定Bean名字,在通知定义时使用pointcut-ref属性通过该id引用切入点,expression属性指定切入点表达式: 

<aop:config>  
   <aop:pointcut id="pointcut" expression="execution(* cn.javass..*.*(..))"/>  
   <aop:aspect ref="aspectSupportBean">  
      <aop:before pointcut-ref="pointcut" method="before"/>  
   </aop:aspect>  
</aop:config>
    2)在<aop:aspect>标签下使用<aop:pointcut>声明一个切入点Bean,该切入点可以被多个切面使用,但一般该切入点只被该切面使用,当然也可以被其他切面使用,但最好不要那样使用,该切入点使用id属性指定Bean名字,在通知定义时使用pointcut-ref属性通过该id引用切入点,expression属性指定切入点表达式:
<aop:config>  
   <aop:aspect ref="aspectSupportBean">  
      <aop:pointcut id=" pointcut" expression="execution(* cn.javass..*.*(..))"/>  
      <aop:before pointcut-ref="pointcut" method="before"/>  
   </aop:aspect>  
</aop:config> 
    3)匿名切入点Bean,可以在声明通知时通过pointcut属性指定切入点表达式,该切入点是匿名切入点,只被该通知使用: 
<aop:config>  
   <aop:aspect ref="aspectSupportBean">  
       <aop:after pointcut="execution(* cn.javass..*.*(..))" method="afterFinallyAdvice"/>  
   </aop:aspect>  
</aop:config>

2.3  声明通知

基于Schema方式支持前边介绍的5中通知类型: 

一、前置通知:在切入点选择的方法之前执行,通过<aop:aspect>标签下的<aop:before>标签声明:

<aop:before pointcut="切入点表达式"  pointcut-ref="切入点Bean引用"  method="前置通知实现方法名"  
arg-names="前置通知实现方法参数列表参数名字"/>   

<bean id="helloWorldService" class="cn.javass.spring.chapter6.service.impl.HelloWorldService"/>  
<bean id="aspect" class="cn.javass.spring.chapter6.aop.HelloWorldAspect"/>  
<aop:config>  
    <aop:aspect ref="aspect">  
        <aop:before pointcut="execution(* cn.javass..*.sayBefore(..)) and args(param)"   
                           method="beforeAdvice(java.lang.String)"   
                           arg-names="param"/>  
    </aop:aspect>  
</aop:config>

pointcut和pointcut-ref:二者选一,指定切入点;

method:指定前置通知实现方法名,如果是多态需要加上参数类型,多个用“,”隔开,如beforeAdvice(java.lang.String);

arg-names:指定通知实现方法的参数名字,多个用“,”分隔,可选


分析一下吧:
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>标签声明: 

<aop:after-returning pointcut="切入点表达式"  pointcut-ref="切入点Bean引用"  
    method="后置返回通知实现方法名"  
    arg-names="后置返回通知实现方法参数列表参数名字"  
    returning="返回值对应的后置返回通知实现方法参数名"  
/> 


<aop:after-returning pointcut="execution(* cn.javass..*.sayAfterReturning(..))"  
                                method="afterReturningAdvice"  
                               arg-names="retVal"    
                               returning="retVal"/> 
pointcut和pointcut-ref:同前置通知同义;
method:同前置通知同义;
arg-names:同前置通知同义;
returning:定义一个名字,该名字用于匹配通知实现方法的一个参数名,当目标方法执行正常返回后,将把目标方法返回值传给通知方法;returning限定了只有目标方法返回值匹配与通知方法相应参数类型时才能执行后置返回通知,否则不执行,对于returning对应的通知方法参数为Object类型将匹配任何目标返回值。

分析一下吧:
1)切入点匹配:在配置中使用“execution(* cn.javass..*.sayAfterReturning(..)) ”匹配目标方法sayAfterReturning,该方法返回true;
2)目标方法定义:使用method="afterReturningAdvice"指定后置返回通知实现方法;
3)目标方法参数命名:其中使用arg-names="retVal"指定通知实现方法参数名为“retVal”;
4)返回值命名:returning="retVal"用于将目标返回值赋值给通知实现方法参数名为“retVal”的参数上。


三、后置异常通知:在切入点选择的方法抛出异常时执行,通过<aop:aspect>标签下的<aop:after-throwing>标签声明:

<aop:after-throwing pointcut="切入点表达式"  pointcut-ref="切入点Bean引用"  
                                method="后置异常通知实现方法名"  
                                arg-names="后置异常通知实现方法参数列表参数名字"  
                                throwing="将抛出的异常赋值给的通知实现方法参数名"/> 

<aop:after-throwing pointcut="execution(* cn.javass..*.sayAfterThrowing(..))"  
                                method="afterThrowingAdvice"  
                                arg-names="exception"  
                                throwing="exception"/>
pointcut和pointcut-ref:同前置通知同义;
method:同前置通知同义;
arg-names:同前置通知同义;
throwing:定义一个名字,该名字用于匹配通知实现方法的一个参数名,当目标方法抛出异常返回后,将把目标方法抛出的异常传给通知方法;throwing限定了只有目标方法抛出的异常匹配与通知方法相应参数异常类型时才能执行后置异常通知,否则不执行,对于throwing对应的通知方法参数为Throwable类型将匹配任何异常。


分析一下吧:
1)切入点匹配:在配置中使用“execution(* cn.javass..*.sayAfterThrowing(..))”匹配目标方法sayAfterThrowing,该方法将抛出RuntimeException异常;
2)目标方法定义:使用method="afterThrowingAdvice"指定后置异常通知实现方法;
3)目标方法参数命名:其中使用arg-names="exception"指定通知实现方法参数名为“exception”;
4)异常命名:returning="exception"用于将目标方法抛出的异常赋值给通知实现方法参数名为“exception”的参数上。


四、后置最终通知:在切入点选择的方法返回时执行,不管是正常返回还是抛出异常都执行,通过<aop:aspect>标签下的<aop:after >标签声明:

<aop:after pointcut="切入点表达式"  pointcut-ref="切入点Bean引用"  
                  method="后置最终通知实现方法名"  
                  arg-names="后置最终通知实现方法参数列表参数名字"/>


<aop:after pointcut="execution(* cn.javass..*.sayAfterFinally(..))"  
         method="afterFinallyAdvice"/> 
pointcut和pointcut-ref:同前置通知同义;
method:同前置通知同义;
arg-names:同前置通知同义;

分析一下吧:
1)切入点匹配:在配置中使用“execution(* cn.javass..*.sayAfterFinally(..))”匹配目标方法sayAfterFinally,该方法将抛出RuntimeException异常;
2)目标方法定义:使用method=" afterFinallyAdvice "指定后置最终通知实现方法。


五、环绕通知:环绕着在切入点选择的连接点处的方法所执行的通知,环绕通知非常强大,可以决定目标方法是否执行,什么时候执行,执行时是否需要替换方法参数,执行完毕是否需要替换返回值,可通过<aop:aspect>标签下的<aop:around >标签声明:

<aop:around pointcut="切入点表达式"  pointcut-ref="切入点Bean引用"  
                     method="后置最终通知实现方法名"  
                     arg-names="后置最终通知实现方法参数列表参数名字"/>  



首先在cn.javass.spring.chapter6.service.IhelloWorldService定义一个测试方法:
public void sayAround(String param);  


其次在cn.javass.spring.chapter6.service.impl. HelloWorldService定义实现
@Override  
public void sayAround(String param) {  
   System.out.println("============around param:" + param);  
} 


第三在cn.javass.spring.chapter6.aop. HelloWorldAspect定义通知实现:
public Object aroundAdvice(ProceedingJoinPoint pjp) throws Throwable {  
    System.out.println("===========around before advice");  
    Object retVal = pjp.proceed(new Object[] {"replace"});  
    System.out.println("===========around after advice");  
    return retVal;  
} 


最后在chapter6/advice.xml配置文件中接着前置通知配置的例子添加如下配置:
<aop:around pointcut="execution(* cn.javass..*.sayAround(..))"  
           method="aroundAdvice"/> 


测试代码cn.javass.spring.chapter6.AopTest:
@Test  
public void testSchemaAroundAdvice() {  
    System.out.println("======================================");  
    ApplicationContext ctx = new ClassPathXmlApplicationContext("chapter6/advice.xml");  
    IHelloWorldService helloworldService =  
    ctx.getBean("helloWorldService", IHelloWorldService.class);  
    helloworldService.sayAround("haha");  
    System.out.println("======================================");  
}  

将输出:	
======================================
===========around before advice
============around param:replace
===========around after advice
======================================
pointcut和pointcut-ref:同前置通知同义;
method:同前置通知同义;
arg-names:同前置通知同义;
 
环绕通知第一个参数必须是org.aspectj.lang.ProceedingJoinPoint类型,在通知实现方法内部使用ProceedingJoinPoint的proceed()方法使目标方法执行,proceed 方法可以传入可选的Object[]数组,该数组的值将被作为目标方法执行时的参数。

分析一下吧:
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”,说明参数被替换了。


2.4 引入

Spring引入允许为目标对象引入新的接口,即可因此引入新功能。通过在<aop:aspect>标签内使用< aop:declare-parents>标签进行引入,定义方式如下:

<aop:declare-parents  
          types-matching="AspectJ语法类型表达式"  
          implement-interface=引入的接口"               
          default-impl="引入接口的默认实现"  
          delegate-ref="引入接口的默认实现Bean引用"/> 


首先定义引入的接口及默认实现:

package cn.javass.spring.chapter6.service;  
public interface IIntroductionService {  
    public void induct();  
} 

package cn.javass.spring.chapter6.service.impl;  
import cn.javass.spring.chapter6.service.IIntroductionService;  
public class IntroductiondService implements IIntroductionService {  
    @Override  
    public void induct() {  
        System.out.println("=========introduction");  
    }  
} 


其次在chapter6/advice.xml配置文件中接着前置通知配置的例子添加如下配置:

<aop:declare-parents  
    types-matching="cn.javass..*.IHelloWorldService+"  
    implement-interface="cn.javass.spring.chapter6.service.IIntroductionService"               
    default-impl="cn.javass.spring.chapter6.service.impl.IntroductiondService"/>


最后测试一下吧,测试代码cn.javass.spring.chapter6.AopTest:
@Test  
public void testSchemaIntroduction() {  
    System.out.println("======================================");  
    ApplicationContext ctx = new ClassPathXmlApplicationContext("chapter6/advice.xml");  
    IIntroductionService introductionService =  
    ctx.getBean("helloWorldService", IIntroductionService.class);  
    introductionService.induct();  
    System.out.println("======================================");  
}  
  
将输出:	
======================================
=========introduction
======================================

types-matching:匹配需要引入接口的目标对象的AspectJ语法类型表达式;
implement-interface:定义需要引入的接口;
default-impl和delegate-ref:定义引入接口的默认实现,二者选一,default-impl是接口的默认实现类全限定名,而delegate-ref是默认的实现的委托Bean名;
 
分析一下吧:
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);”可直接获取到引入的接口。

<aop:advisor pointcut="切入点表达式" pointcut-ref="切入点Bean引用"  
                     advice-ref="通知API实现引用"/>  


首先在cn.javass.spring.chapter6.service.IhelloWorldService定义一个测试方法:
public void sayAdvisorBefore(String param); 


其次在cn.javass.spring.chapter6.service.impl. HelloWorldService定义实现
@Override  
public void sayAdvisorBefore(String param) {  
    System.out.println("============say " + param);  
} 


第三定义前置通知API实现:
package cn.javass.spring.chapter6.aop;  
import java.lang.reflect.Method;  
import org.springframework.aop.MethodBeforeAdvice;  
public class BeforeAdviceImpl implements MethodBeforeAdvice {  
    @Override  
    public void before(Method method, Object[] args, Object target) throws Throwable {  
        System.out.println("===========before advice");  
    }  
} 

在chapter6/advice.xml配置文件中先添加通知实现Bean定义:
<bean id="beforeAdvice" class="cn.javass.spring.chapter6.aop.BeforeAdviceImpl"/>  


然后在<aop:config>标签下,添加Advisor定义,添加时注意顺序:
<aop:advisor pointcut="execution(* cn.javass..*.sayAdvisorBefore(..))"  
                     advice-ref="beforeAdvice"/>  

测试代码cn.javass.spring.chapter6.AopTest:
@Test  
public void testSchemaAdvisor() {  
   System.out.println("======================================");  
   ApplicationContext ctx = new ClassPathXmlApplicationContext("chapter6/advice.xml");  
   IHelloWorldService helloworldService =  
   ctx.getBean("helloWorldService", IHelloWorldService.class);  
   helloworldService.sayAdvisorBefore("haha");  
   System.out.println("======================================");  
}  

将输出:	
======================================
===========before advice
============say haha
======================================
pointcut和pointcut-ref:二者选一,指定切入点表达式;
advice-ref:引用通知API实现Bean,如前置通知接口为MethodBeforeAdvice; 

不推荐使用Advisor,除了在进行事务控制的情况下,其他情况一般不推荐使用该方式,该方式属于侵入式设计,必须实现通知API。



参考来源:

【第六章】 AOP 之 6.2 AOP的HelloWorld ——跟我学spring3

【第六章】 AOP 之 6.3 基于Schema的AOP ——跟我学spring3


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值