Spring的AOP原理

14 篇文章 0 订阅
3 篇文章 0 订阅

一、什么是 AOP

AOP(Aspect-OrientedProgramming,面向切面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。当我们需要为分散的对象引入公共行为(日志、安全、事务)的时候,OOP则显得无能为力。也就是说,OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系。例如日志功能。日志代码往往水平地散布在所有对象层次中,而与它所散布到的对象的核心功能毫无关系。对于其他类型的代码,如安全性、异常处理和透明的持续性也是如此。这种散布在各处的无关的代码被称为横切(cross-cutting)代码,在OOP设计中,它导致了大量代码的重复,模块间的藕合度高,而不利于各个模块的重用。

 

而AOP技术则恰恰相反,它利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为“Aspect”,即切面。所谓“切面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块间的耦合度,有利于未来的可操作性和可维护性。AOP代表的是一个横向的关系,如果说“对象”是一个空心的圆柱体,其中封装的是对象的属性和行为;那么面向方面编程的方法,就仿佛一把利刃,将这些空心圆柱体剖开,以获得其内部的消息。而剖开的切面,也就是所谓的“方面”了。然后它又以巧夺天功的妙手将这些剖开的切面复原,不留痕迹。

 

使用“横切”技术,AOP把软件系统分为两个部分:核心业务逻辑组件和横切关注点。横切关注点模块化为特殊的类,这些类被称为“切面”,好处:1.横切关注点都集中于一块,不会出现大量重复代码;2.核心模块只关注核心功能的代码,模块间藕合度降低。

二、AOP 的实现原理

<img src="AOP代理.jpg" />

如图:AOP 实际上是由目标类的代理类实现的AOP 代理其实是由 AOP 框架动态生成的一个对象,该对象可作为目标对象使用。AOP 代理包含了目标对象的全部方法,但 AOP 代理中的方法与目标对象的方法存在差异,AOP 方法在特定切入点添加了增强处理,并回调了目标对象的方法

三、AOP相关概念

连接点(Joinpoint:在程序执行过程中某个特定的点,比如某方法调用的时候或者处理异常的时候。在Spring AOP中,一个连接点总是表示一个方法的执行。通俗的说就是加入切点的那个点

通知(Advice:在切面的某个特定的连接点上执行的动作。其中包括了“around”“before”“after”等不同类型的通知(通知的类型将在后面部分进行讨论)。许多AOP框架(包括Spring)都是以拦截器做通知模型,并维护一个以连接点为中心的拦截器链。

切入点(Pointcut:匹配连接点的断言。通知和一个切入点表达式关联,并在满足这个切入点的连接点上运行(例如,当执行某个特定名称的方法时)。切入点表达式如何和连接点匹配是AOP的核心:Spring缺省使用AspectJ切入点语法。

引入(Introduction:用来给一个类型声明额外的方法或属性(也被称为连接类型声明(inter-type declaration))。Spring允许引入新的接口(以及一个对应的实现)到任何被代理的对象。例如,你可以使用引入来使一个bean实现IsModified接口,以便简化缓存机制。

织入(Weaving:将切面应用到目标对象来创建新的代理对象的过程。这些可以在编译时(例如使用AspectJ编译器),类加载时和运行时完成。Spring和其他纯Java AOP框架一样,在运行时完成织入。

增强(Advice):是织入到目标类连接点上的一段程序代码。Spring使用增强类定义横切逻辑,同时由于Spring只支持方法连接点,增强还包括了在方法上的哪一点加入横切代码的方位信息,所以增强既包括横切逻辑、还包含部分连接点的信息。

引介(Introduction):是一种特殊的增强,为类添加一些属性和方法。

切面(Advisor):代表一般切面,包含了横切代码和连接点信息,本身是一个简单的切面,横切的连接点是目标类的所有方法。3种类型:一般切面(advisor)、切点切面(PointcutAdvisor)、引介切面(IntroductionAdvisor)。


四、Spring中AOP的实现

第一种是基于xml配置文件方式的实现,

第二种是基于注解方式的实现。

首先我们来看一下业务逻辑service层:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
/** 
  * RegisterService的实现类 
  * @author 曹胜欢 */  
public  class  RegisterServiceImpl  implements  RegisterService {  
     private   RegisterDao registerDao;  
     public  RegisterServiceImpl() {}  
     /** 带参数的构造方法 */  
     public  RegisterServiceImpl(RegisterDao  registerDao){  
         this .registerDao =registerDao;  
     }  
     public  void  save(String loginname, String password) {  
         registerDao.save(loginname, password);  
         throw  new  RuntimeException( "故意抛出一个异常。。。。" );  
     }  
       /** set方法 */  
     public  void  setRegisterDao(RegisterDao registerDao) {  
         this .registerDao = registerDao;  
}}

对于业务系统来说,RegisterServiceImpl类就是目标实现类,它的业务方法,如save()方法的前后或代码会出现异常的地方都是AOP的连接点。

 

下面是日志服务类的代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
/** 
  * 日志切面类 
  * @author 曹胜欢 
  */  
public  class  LogAspect {  
     //任何通知方法都可以将第一个参数定义为 org.aspectj.lang.JoinPoint类型   
     public  void  before(JoinPoint call) {  
         //获取目标对象对应的类名  
         String className = call.getTarget().getClass().getName();  
         //获取目标对象上正在执行的方法名  
         String methodName = call.getSignature().getName();  
         System.out.println( "前置通知:"  + className +  "类的"  + methodName +  "方法开始了" );  
     }  
     public  void  afterReturn() {  
         System.out.println( "后置通知:方法正常结束了" );  
     }  
     public  void  after(){  
         System.out.println( "最终通知:不管方法有没有正常执行完成,一定会返回的" );  
     }  
     public  void  afterThrowing() {  
         System.out.println( "异常抛出后通知:方法执行时出异常了" );  
     }  
     //用来做环绕通知的方法可以第一个参数定义为org.aspectj.lang.ProceedingJoinPoint类型  
     public  Object doAround(ProceedingJoinPoint call)  throws  Throwable {  
         Object result =  null ;  
         this .before(call); //相当于前置通知  
         try  {  
             result = call.proceed();  
             this .afterReturn();  //相当于后置通知  
         catch  (Throwable e) {  
             this .afterThrowing();   //相当于异常抛出后通知  
             throw  e;  
         } finally {  
             this .after();   //相当于最终通知  
         }  
         return  result;  
     }  
}

  这个类属于业务服务类,如果用AOP的术语来说,它就是一个切面类,它定义了许多通知。Before()afterReturn()after()afterThrowing()这些方法都是通知。

 

下面我们就来看具体配置,首先来看一下:

<1>.基于xml配置文件的AOP实现:这种方式在实现AOP时,有4个步骤

?
1
< br >


?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
<? 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-2.5.xsd  
             http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd>  
     < bean  id = "registerDaoImpl"  class = "com.zxf.dao.RegisterDaoImpl" />  
     < bean  id = "registerService"  class = "com.zxf.service.RegisterServiceImpl" >  
         < property  name = " registerDaoImpl "  ref = " RegisterDaoImpl " />  
     </ bean >  
     <!-- 日志切面类 -->  
     < bean  id = "logAspectBean"  class = "com.zxf.aspect.LogAspect" />  
     <!-- 第1步: AOP的配置 -->  
     < aop:config >  
         <!-- 第2步:配置一个切面 -->  
         < aop:aspect  id = "logAspect"  ref = "logAspectBean" >  
             <!-- 第3步:定义切入点,指定切入点表达式 -->  
             < aop:pointcut  id = "allMethod"   
                 expression = "execution(* com.zxf.service.*.*(..))" />   
             <!-- 第4步:应用前置通知 -->  
             < aop:before  method = "before"  pointcut-ref = "allMethod"  />  
             <!-- 第4步:应用后置通知 -->  
             < aop:after-returning  method = "afterReturn"  pointcut-ref = "allMethod" />  
             <!-- 第4步:应用最终通知 -->  
             < aop:after  method = "after"  pointcut-ref = "allMethod" />  
             <!-- 第4步:应用抛出异常后通知 -->  
             < aop:after-throwing  method = "afterThrowing"  pointcut-ref = "allMethod" />  
             <!-- 第4步:应用环绕通知 -->  
             <!--  
             <aop:around method="doAround" pointcut-ref="allMethod" /> 
              -->  
         </ aop:aspect >  
     </ aop:config >  
</ beans >

上述配置针对切入点应用了前置、后置、最终,以及抛出异常后通知。这样在测试执行RegisterServiceImpl类的save()方法时,控制台会有如下结果输出:

 

前置通知:com.zxf.service.RegisterServiceImpl类的save方法开始了。

针对MySQL的RegisterDao实现中的save()方法。

后置通知:方法正常结束了。

最终通知:不管方法有没有正常执行完成,一定会返回的。

下面我们在来看一下第二种配置方式:

<2>基于注解的AOP的实现

 

     首先创建一个用来作为切面的类LogAnnotationAspect,同时把这个类配置在spring的配置文件中。

        在spring2.0以后引入了JDK5.0的注解Annotation的支持,提供了对AspectJ基于注解的切面的支持,从而 更进一步地简化AOP的配置。具体的步骤有两步。

 

Spring的配置文件是如下的配置:

 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
<? 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-2.5.xsd  
             http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd>  
     < bean  id = "registerDao"  class = "com.zxf.dao.RegisterDaoImpl" />  
     < bean  id = "registerService"  class = "com.zxf.service.RegisterServiceImpl" >  
         < property  name = "registerDao"  ref = "registerDao" />  
     </ bean >  
     <!-- 把切面类交由Spring容器来管理 -->  
     < bean  id = "logAspectBean"  class = "com.zxf.aspect.LogAnnotationAspect" />  
     <!-- 启用spring对AspectJ注解的支持 -->  
     < aop:aspectj-autoproxy />  
</ beans >

这是那个切面的类LogAnnotationAspect

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
/** 
  * 日志切面类 
  */  
@Aspect   //定义切面类  
public  class  LogAnnotationAspect {  
     @SuppressWarnings ( "unused" )  
     //定义切入点,提供一个方法,这个方法的名字就是改切入点的id  
     @Pointcut ( "execution(* com.zxf.service.*.*(..))" )  
     private  void  allMethod(){}  
     //针对指定的切入点表达式选择的切入点应用前置通知  
     @Before ( "execution(* com. zxf.service.*.*(..))" )  
     public  void  before(JoinPoint call) {  
         String className = call.getTarget().getClass().getName();  
         String methodName = call.getSignature().getName();  
         System.out.println( "【注解-前置通知】:"  + className +  "类的"   
                 + methodName +  "方法开始了" );  
     }  
     //访问命名切入点来应用后置通知  
     @AfterReturning ( "allMethod()" )  
     public  void  afterReturn() {  
         System.out.println( "【注解-后置通知】:方法正常结束了" );  
     }  
     //应用最终通知  
     @After ( "allMethod()" )  
     public  void  after(){  
         System.out.println( "【注解-最终通知】:不管方法有没有正常执行完成,"   
                 "一定会返回的" );  
     }  
     //应用异常抛出后通知  
     @AfterThrowing ( "allMethod()" )  
     public  void  afterThrowing() {  
         System.out.println( "【注解-异常抛出后通知】:方法执行时出异常了" );  
     }  
     //应用周围通知  
     //@Around("allMethod()")  
     public  Object doAround(ProceedingJoinPoint call)  throws  Throwable{  
         Object result =  null ;  
         this .before(call); //相当于前置通知  
         try  {  
             result = call.proceed();  
             this .afterReturn();  //相当于后置通知  
         catch  (Throwable e) {  
             this .afterThrowing();   //相当于异常抛出后通知  
             throw  e;  
         } finally {  
             this .after();   //相当于最终通知  
         }  
         return  result;  
     }  
}

五:Spring中AOP的两种代理方式(Java动态代理和CGLIB代理)

Spring AOP使用动态代理技术在运行期织入增强代码。使用两种代理机制:基于JDK的动态代理(JDK本身只提供接口的代理);基于CGlib的动态代理

1. JDK的动态代理主要涉及java.lang.reflect包中的两个类:Proxy和InvocationHandler。其中InvocationHandler只是一个接口,可以通过实现该接口定义横切逻辑,并通过反射机制调用目标类的代码,动态的将横切逻辑与业务逻辑织在一起。而Proxy利用InvocationHandler动态创建一个符合某一接口的实例,生成目标类的代理对象。(只能为接口创建代理实例)

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
//需要被代理的接口
public  interface  ForumService {
     public  void  removeTopic( int  topicId);
     public  void  removeForum( int  forumId);
}
 
//被代理接口的实现类,包含核心的业务逻辑
public  class  ForumServiceImpl  implements  ForumService{
     @Override
     public  void  removeTopic( int  topicId) {
         System.out.println( "模拟删除Topic记录:" + topicId);
         try  {
             Thread.currentThread().sleep( 20 );
         catch  (InterruptedException e) {
             e.printStackTrace();
         }
     }
 
     @Override
     public  void  removeForum( int  forumId) {
         System.out.println( "模拟删除Forum记录:" + forumId);
         try  {
             Thread.currentThread().sleep( 40 );
         catch  (InterruptedException e) {
             e.printStackTrace();
         }
     }
     
}
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
//性能监控核心代码生成
public  class  MethodPerformance {
     private  long  begin;
     private  long  end;
     private  String serviceMethod;
     
     public  MethodPerformance(String serviceMethod) {
         this .serviceMethod = serviceMethod;
         this .begin = System.currentTimeMillis();
     };
     
     public  void  printPerformance(){
         this .end = System.currentTimeMillis();
         long  elapse =  this .end -  this .begin;
         System.out.println(serviceMethod +  " cost "  + elapse + "ms" );
     }
}
 
//线程安全的横切逻辑
public  class  PerformanceMonitor {
     private  static  ThreadLocal<MethodPerformance> tl=  new  ThreadLocal<MethodPerformance>();
     
     public  static  void  begin(String method){
         System.out.println( "begin monitor" );
         MethodPerformance mp =  new  MethodPerformance(method);
         tl.set(mp);
     }
     
     public  static  void  end(){
         System.out.println( "end monitor" );
         MethodPerformance mp = tl.get();
         mp.printPerformance();
     }
}
 
//AOP横切模块
public  class  PerfermanceHandler  implements  InvocationHandler{
     private  Object target;
     
     public  PerfermanceHandler(Object target) {
         this .target = target;
     }
 
     @Override
     public  Object invoke(Object proxy, Method method, Object[] args)
             throws  Throwable {
         PerformanceMonitor.begin(target.getClass().getName()+ "." +method.getName());
         Object object = method.invoke(target, args);
         PerformanceMonitor.end();
         return  object;
     }
 
}
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
//测试
public  class  TestForumService {
     public  static  void  main(String[] args) {
         ForumService target =  new  ForumServiceImpl();
         //将目标业务类与横切代码编织到一起
         PerfermanceHandler handler =  new  PerfermanceHandler(target);
         //创建代理实例
         ForumService proxy = (ForumService) Proxy.newProxyInstance(target.getClass().getClassLoader(), 
                                                     target.getClass().getInterfaces(), handler);
         proxy.removeForum( 10 );
         proxy.removeTopic( 1012 );
         
     }
}

begin monitor

模拟删除Forum记录:10

end monitor

com.baobaotao.proxy.ForumServiceImpl.removeForum cost 42ms

begin monitor

模拟删除Topic记录:1012

end monitor

com.baobaotao.proxy.ForumServiceImpl.removeTopic cost 21ms


2.CGLib采用底层的字节码技术,为一个类创建子类,并在子类中采用方法拦截的技术拦截所有父类的调用方法,并顺势织入横切逻辑。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
package  com.baobaotao.proxy;
 
import  java.lang.reflect.Method;
 
import  org.springframework.cglib.proxy.Enhancer;
import  org.springframework.cglib.proxy.MethodInterceptor;
import  org.springframework.cglib.proxy.MethodProxy;
 
public  class  CglibProxy  implements  MethodInterceptor{
 
     private  Enhancer enhancer =  new  Enhancer();
     
     public  Object getProxy(Class clazz){
         enhancer.setSuperclass(clazz); //设置创建子类的类
         enhancer.setCallback( this );
         return  enhancer.create(); //通过字节码技术动态创建子类实例
     }
     
     @Override
     public  Object intercept(Object target, Method method, Object[] args,
             MethodProxy proxy)  throws  Throwable {
         PerformanceMonitor.begin(target.getClass().getName()+ "." +method.getName());
         Object object = proxy.invokeSuper(target, args);
         PerformanceMonitor.end();
         return  object;
     }
}
?
1
2
3
4
5
6
7
8
9
10
11
12
package  com.baobaotao.proxy;
 
import  java.lang.reflect.Proxy;
//测试
public  class  TestForumService {
     public  static  void  main(String[] args) {  
         CglibProxy cglibProxy =  new  CglibProxy();
         ForumServiceImpl service = (ForumServiceImpl) cglibProxy.getProxy(ForumServiceImpl. class );
         service.removeForum( 10 );
         service.removeTopic( 1012 );
     }
}

begin monitor

模拟删除Forum记录:10

end monitor

com.baobaotao.proxy.ForumServiceImpl$$EnhancerByCGLIB$$626e31f5(子类).removeForum cost 67ms

begin monitor

模拟删除Topic记录:1012

end monitor

com.baobaotao.proxy.ForumServiceImpl$$EnhancerByCGLIB$$626e31f5.removeTopic cost 20ms


Spring增强类

Spring支持5种增强类型:

1)前置增强:org.springframework.aop.BeforeAdvice代表前置增强,spring只支持方法级的增强,目前可用MethodBeforeAdvice。

?
1
2
3
4
public  interface  Waiter {
     public  void  greetTo(String name);
     public  void  serveTo(String name);
}
?
1
2
3
4
5
6
7
8
9
10
11
12
public  class  NaiveWaiter  implements  Waiter {
 
     @Override
     public  void  greetTo(String name) {
         System.out.println( "greetTo " + name + "..." );
     }
 
     @Override
     public  void  serveTo(String name) {
         System.out.println( "serveTo " + name + "..." );    
     }
}
?
1
2
3
4
5
6
7
8
9
10
11
12
import  java.lang.reflect.Method;
import  org.springframework.aop.MethodBeforeAdvice;
 
public  class  GreeteBeforeAdvice  implements  MethodBeforeAdvice{
 
     @Override
     public  void  before(Method arg0, Object[] arg1, Object arg2)
             throws  Throwable {
         String client = (String) arg1[ 0 ];
         System.out.println( "How are you!" +client+ "." );
     }
}
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
import  org.springframework.aop.BeforeAdvice;
import  org.springframework.aop.framework.ProxyFactory;
 
public  class  TestBeforeAdvice {
 
     public  static  void  main(String[] args) {
         Waiter target =  new  NaiveWaiter();
         BeforeAdvice advice =  new  GreeteBeforeAdvice();
         //Spring提供的代理工厂
         ProxyFactory pf =  new  ProxyFactory(); //使用Cglib2AOPProx 即CGlib代理技术创建代理
//      pf.setInterfaces(target.getClass().getInterfaces());//使用JdkDynamicAopProxy
         //设置代理目标
         pf.setTarget(target);
         //添加增强处理
         pf.addAdvice(advice);
         //生成代理实例
         Waiter waiter = (Waiter) pf.getProxy();
         waiter.greetTo( "Tom" );
         waiter.serveTo( "Lucy" );
         
//      ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");
//      Waiter waiter1 = (Waiter) ctx.getBean("waiter");
//      waiter1.greetTo("John");       
     }
}
?
1
2
3
4
5
< bean  id = "target"  class = "com.baobaotao.advice.NaiveWaiter"  />
< bean  id = "greeteAdvice"  class = "com.baobaotao.advice.GreeteBeforeAdvice"  />
< bean  id = "waiter"  class = "org.springframework.aop.framework.ProxyFactoryBean"
     p:proxyInterfaces = "com.baobaotao.advice.Waiter"  p:interceptorNames = "greeteAdvice"
     p:target-ref = "target"  />

2)后置增强:org.springframework.aop.AfterReturningAdvice代表后置增强,在目标方法执行后实施增强。

?
1
2
3
4
5
6
7
8
9
import  java.lang.reflect.Method;
import  org.springframework.aop.AfterReturningAdvice;
public  class  GreeteAfterAdvice  implements  AfterReturningAdvice{
     @Override
     public  void  afterReturning(Object arg0, Method arg1, Object[] arg2,
             Object arg3)  throws  Throwable {
         System.out.println( "Please enjoy yourself!" );
     }
}
?
1
2
3
< bean  id = "waiter"  class = "org.springframework.aop.framework.ProxyFactoryBean"
         p:proxyInterfaces = "com.baobaotao.advice.Waiter"  p:interceptorNames = "greeteBefore,greeteAfter"
         p:target-ref = "target"  />

3)环绕增强:org.aopalliance.intercept.MethodInterceptor代表环绕增强,在目标方法执行前后实施增强。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
import  org.aopalliance.intercept.MethodInterceptor;
import  org.aopalliance.intercept.MethodInvocation;
 
public  class  GreeteInterceptor  implements  MethodInterceptor{
 
     @Override
     public  Object invoke(MethodInvocation inv)  throws  Throwable {
         Object[] args = inv.getArguments();
         String client = (String) args[ 0 ];
         System.out.println( "How are you!" +client+ "." );
         Object obj = inv.proceed();
         System.out.println( "Please enjoy yourself!" );
         return  obj;
     }
 
}

4)异常抛出增强:org.springframework.aop.ThrowsAdvice,在目标方法执行抛出异常后实施增强。方法名必须为afterThrowing,如参前三个可选,最后一个是Throwable或其子类。

?
1
2
3
4
5
6
7
8
9
10
11
import  java.lang.reflect.Method;
import  org.springframework.aop.ThrowsAdvice;
public  class  TransactionManager  implements  ThrowsAdvice {
     public  void  afterThrowing(Method method, Object[] args, Object target,
             Exception ex)  throws  Throwable {
         System.out.println( "-----------" );
         System.out.println( "method:"  + method.getName());
         System.out.println( "抛出异常:"  + ex.getMessage());
         System.out.println( "成功回滚事务。" );
     }
}

5)引介增强:org.springframework.aop.IntroductionInterceptor,表示目标类添加一些新的方法和属性,连接点是类级别,而不是方法级别。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
import  org.aopalliance.intercept.MethodInvocation;
import  org.springframework.aop.support.DelegatingIntroductionInterceptor;
 
import  com.baobaotao.proxy.PerformanceMonitor;
 
public  class  ControllablePerformaceMonitor  extends  DelegatingIntroductionInterceptor  implements  Monitorable{
     private  ThreadLocal<Boolean> MonitorStatusMap =  new  ThreadLocal<Boolean>();
     @Override
     public  void  setMonitorActive( boolean  active) {
         MonitorStatusMap.set(active);
     }
     @Override
     public  Object invoke(MethodInvocation mi)  throws  Throwable {
         Object obj =  null ;
         if  (MonitorStatusMap.get() !=  null  && MonitorStatusMap.get()) {
             PerformanceMonitor.begin(mi.getClass().getName() +  "."
                     + mi.getMethod().getName());
             obj =  super .invoke(mi);
             PerformanceMonitor.end();
         else  {
             obj =  super .invoke(mi);
         }
         return  obj;
     }
 
}
?
1
2
3
4
5
< bean  id = "forumService"  class = "org.springframework.aop.framework.ProxyFactoryBean"
         p:interfaces = "com.baobaotao.introduce.Monitorable"  //引入引介增强要实现的接口
         p:target-ref = "forumServiceTarget"
         p:interceptorNames = "pmonitor" 
         p:proxyTargetClass = "true"  />//只能通过为目标类型类创建子类的方式生成增强的代理
AOP切面的配置方式

1)基于Schema的配置:在xml中描述切点、增强类型,切面类为pojo

2)基于AspectJ的配置:切点、增强类型使用注解进行描述


转载:http://my.oschina.net/elain/blog/382494

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring AOP(Aspect-Oriented Programming)是一种基于面向切面编程的技术,它通过将一个应用程序分解成许多独立的部分,从而提高了应用程序的模块化程度,同时也提高了代码的可重用性和可维护性。 Spring AOP 是在运行时对程序进行修改的,它不需要重新编译源代码,因此使得代码的维护和修改更加方便。Spring AOP 原理主要分为以下几个方面: 1. 切面(Aspect):切面是一个类,其中包含了一组相关的Advice和其他相关的代码,用于在目标对象的方法执行前、执行后或者抛出异常时执行相应的操作。 2. 连接点(Join point):连接点是指在应用程序的执行过程中,可以被切面拦截的点,比如方法调用、异常抛出、属性赋值等。 3. 通知(Advice):通知是指在连接点处执行的代码,有多种类型的通知,包括前置通知(Before advice)、后置通知(After advice)、返回通知(After returning advice)、异常通知(After throwing advice)和环绕通知(Around advice)。 4. 切入点(Pointcut):切入点是一个表达式,用于定义哪些连接点会被切面拦截。 5. 代理(Proxy):代理是指在目标对象和切面之间创建的对象,它可以拦截目标对象的方法调用,并在方法执行前后执行相应的通知。 Spring AOP 的实现机制主要是通过 JDK 动态代理或者 CGLIB 动态代理来实现的。当目标对象实现了接口时,Spring AOP 使用 JDK 动态代理,否则使用 CGLIB 动态代理。在 JDK 动态代理中,代理对象实现了与目标对象相同的接口,而在 CGLIB 动态代理中,代理对象是目标对象的子类。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值