AOP实现(二)——Spring 2.0中的AOP实现
2012-05-02 20:45
411 查看
在Spring 2.0中,除了传统的通过实现AOP
AIP的方式来实现Advice之外,还提供了两种更加简便的方式来实现Advice:1)基于XML Schema的设置;2)基于Annotation的支持,采用这两种方式,Advice将不用实现特定的接口。现在让我们来看看如何使用这两种方式来分别实现Before
Advice、After Advice、Around Advice、Throwing Advice。
一、Before Advice:基于XML
Schema
当基于XML Schema实现Before Advice时,你的Advice类不用实现org.springframework.aop.MethodBeforeAdvice接口,例如:
java代码
before方法是在目标对象上的方法被执行前要执行的方法,before方法中的JoinPoint参数是可选项,你可以根据需要决定是否需要JoinPoint参数,通过JoinPoint对象,你可以获得目标对象(getTarget())、目标方法上的参数(getArgs())等信息。
然后在XML中为目标对象指定LogBeforeAdvice代理:
xml 代码
如上所示,在Spring 2.0中要使用基于XML Sechma声明AOP的方式,需要在XML中加入aop的名称空间。当基于XML
Sechma实现AOP时,所有的AOP都是在<aop:config></aop:config>标签中声明的,<aop:aspect></aop:aspect>用于定义Advice实例。<aop:before></aop:before>表示当前实例用于实现Before
Advice;pointcut属性用于指定pointcut表示式,上面的例子表示此Advice将应用于com.savage.aop.MessageSender接口中的任何方法;method属性表示Advice上要调用的方法。
现在调用任何MessageSender接口上的方法之前都会执行LogBeforeAdvice的before方法,例如:
java 代码
二、Before Advice:基于Annotation
使用Annotation来实现Advice,在XML文件上的定义要比基于XML Sechema的方法要简便的多,但在实现Before
Advice类时,则需要使用到@Aspect、@Before标识,并需要引入org.aspectj.lang.annotation包中的类。还以LogBeforeAdvice为例,LogBeforeAdvice类需要改为:
java 代码
通过aop命名空间的<aop:aspectj-autoproxy />声明自动为spring容器中那些配置@aspectJ切面的bean
创建代理,织入切面。当然,spring在内部依旧采用AnnotationAwareAspectJAutoProxyCreator进行自动代理的创建工作,但具体实现的细节已经被<aop:aspectj-autoproxy />隐藏起来了, <aop:aspectj-autoproxy />有一个proxy-target-class属性,默认为false,表示使用jdk动态代理织入增强,当配为<aop:aspectj-autoproxy poxy-target-class="true"/>时,表示使用CGLib动态代理技术织入增强。不过即使proxy-target-class设置为false,如果目标类没有声明接口,则spring将自动使用CGLib动态代理。
如上所示,通过@Aspect将一个类声明为Aspect类,通过@Before将方法声明Before Advice,方法中的JoinPoint同样是可选的。然后在XML文件中做如下定义:
xml 代码
所有基于Annotation实现的Advice,在XML文件中都只要使用<aop:aspectj-autoproxy></aop:aspectj-autoproxy>进行设置就可以了,非常简单。
三、After Advice:基于XML Sechma
和Before Advice一样,基于XML Sechma实现After Returning Advice时,不再需要org.springframework.aop.AfterReturningAdvice接口:
java 代码
然后在XML中做如下设置:
xml 代码
四、After Advice:基于Annotation
和Before Advice相似,使用@AfterReturning来表示After Returning Advice:
java 代码
这里和Before Advice有点不同的是,在定义Poincut表示式时,多了一个returning属性,用于指定目标方法执行完后的返回值。
XML文件中的设置与LogBeforeAdvice的相似(将logBeforeAdvice的定义改为logAfterReturning的定义),不再列举。
五、Around Advice:基于XML Sechma
在Spring 2.0中,Around Advice不用实现org.aoplliance.intercept.MethodInterceptor接口,但Advice的方法必须返回对象,并且必须定义一个ProceedingJoinPoint参数,例如:
java 代码
XML中的设置如下:
xml 代码
六、Around Advice:基于Annotation
和Before Advice相似,使用@Around来表示Around Advice:
java 代码
七、Throw Advice:基于XML Sechma
在Spring 2.0中,Throw Advice不用实现org.springframework.aop.ThrowsAdvice接口,但Advice的方法必须定义Throwable(或其子类)参数,例如:
java 代码
在XML的设置如下:
xml 代码
在<aop:after-throwing></aop:after-throwing>中必须定义throwing属性,指定方法中的throwable参数。Spring将根据异常类型决定是否调用afterThrowing方法。
八、Throw Advice:基于Annotation
java 代码
AIP的方式来实现Advice之外,还提供了两种更加简便的方式来实现Advice:1)基于XML Schema的设置;2)基于Annotation的支持,采用这两种方式,Advice将不用实现特定的接口。现在让我们来看看如何使用这两种方式来分别实现Before
Advice、After Advice、Around Advice、Throwing Advice。
一、Before Advice:基于XML
Schema
当基于XML Schema实现Before Advice时,你的Advice类不用实现org.springframework.aop.MethodBeforeAdvice接口,例如:
java代码
package com.savage.aop; import org.aspectj.lang.JoinPoint; public class LogBeforeAdvice { public void before(JoinPoint joinPoint) { System.out.println("Logging before " + joinPoint.getSignature().getName()); } }
before方法是在目标对象上的方法被执行前要执行的方法,before方法中的JoinPoint参数是可选项,你可以根据需要决定是否需要JoinPoint参数,通过JoinPoint对象,你可以获得目标对象(getTarget())、目标方法上的参数(getArgs())等信息。
然后在XML中为目标对象指定LogBeforeAdvice代理:
xml 代码
<?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.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd"> <bean id="messageSender" class="com.savage.aop.HttpMessageSender"></bean> <bean id="logBeforeAdvice" class="com.savage.aop.LogBeforeAdvice"></bean> <aop:config> <aop:aspect id="logBefore" ref="logBeforeAdvice"> <aop:before pointcut="execution(* com.savage.aop.MessageSender.*(..))" method="before"/> </aop:aspect> </aop:config> </beans>
如上所示,在Spring 2.0中要使用基于XML Sechma声明AOP的方式,需要在XML中加入aop的名称空间。当基于XML
Sechma实现AOP时,所有的AOP都是在<aop:config></aop:config>标签中声明的,<aop:aspect></aop:aspect>用于定义Advice实例。<aop:before></aop:before>表示当前实例用于实现Before
Advice;pointcut属性用于指定pointcut表示式,上面的例子表示此Advice将应用于com.savage.aop.MessageSender接口中的任何方法;method属性表示Advice上要调用的方法。
现在调用任何MessageSender接口上的方法之前都会执行LogBeforeAdvice的before方法,例如:
java 代码
package com.savage.aop; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class AdviceDemo { public static void main(String[] args) { ApplicationContext context = new ClassPathXmlApplicationContext("beans-config.xml"); MessageSender sender = (MessageSender)context.getBean("messageSender"); sender.sendMessage("message"); } }
二、Before Advice:基于Annotation
使用Annotation来实现Advice,在XML文件上的定义要比基于XML Sechema的方法要简便的多,但在实现Before
Advice类时,则需要使用到@Aspect、@Before标识,并需要引入org.aspectj.lang.annotation包中的类。还以LogBeforeAdvice为例,LogBeforeAdvice类需要改为:
java 代码
package com.savage.aop; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; @Aspect public class LogBeforeAdvice { @Before("execution(* com.savage.aop.MessageSender.*(..))") public void before(JoinPoint joinPoint) { System.out.println("Logging before " + joinPoint.getSignature().getName()); } }
通过aop命名空间的<aop:aspectj-autoproxy />声明自动为spring容器中那些配置@aspectJ切面的bean
创建代理,织入切面。当然,spring在内部依旧采用AnnotationAwareAspectJAutoProxyCreator进行自动代理的创建工作,但具体实现的细节已经被<aop:aspectj-autoproxy />隐藏起来了, <aop:aspectj-autoproxy />有一个proxy-target-class属性,默认为false,表示使用jdk动态代理织入增强,当配为<aop:aspectj-autoproxy poxy-target-class="true"/>时,表示使用CGLib动态代理技术织入增强。不过即使proxy-target-class设置为false,如果目标类没有声明接口,则spring将自动使用CGLib动态代理。
如上所示,通过@Aspect将一个类声明为Aspect类,通过@Before将方法声明Before Advice,方法中的JoinPoint同样是可选的。然后在XML文件中做如下定义:
xml 代码
<?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.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd"> <bean id="messageSender" class="com.savage.aop.HttpMessageSender"></bean> <bean id="logBeforeAdvice" class="com.savage.aop.LogBeforeAdvice"></bean> <aop:aspectj-autoproxy/> </beans>
所有基于Annotation实现的Advice,在XML文件中都只要使用<aop:aspectj-autoproxy></aop:aspectj-autoproxy>进行设置就可以了,非常简单。
三、After Advice:基于XML Sechma
和Before Advice一样,基于XML Sechma实现After Returning Advice时,不再需要org.springframework.aop.AfterReturningAdvice接口:
java 代码
package com.savage.aop; import org.aspectj.lang.JoinPoint; public class LogAfterReturningAdvice { public void afterReturning(JoinPoint joinPoint) { System.out.println("Logging after " + joinPoint.getSignature().getName()); } }
然后在XML中做如下设置:
xml 代码
<?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.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd"> <bean id="messageSender" class="com.savage.aop.HttpMessageSender"></bean> <bean id="logAfterReturningAdvice" class="com.savage.aop.LogAfterReturningAdvice"></bean> <aop:config> <aop:aspect id="logAfterReturning" ref="logAfterReturningAdvice"> <aop:after-returning pointcut="execution(* com.savage.aop.MessageSender.*(..))" method="afterReturning"/> </aop:aspect> </aop:config> </beans>
四、After Advice:基于Annotation
和Before Advice相似,使用@AfterReturning来表示After Returning Advice:
java 代码
package com.savage.aop; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.AfterReturning; @Aspect public class AfterReturningAdvice { @AfterReturning(pointcut="execution(* com.savage.aop.MessageSender.*(..))", returning="retVal") public void afterReturning(JoinPoint joinPoint, Object retVal) { System.out.println("Logging after " + joinPoint.getSignature().getName()); } }
这里和Before Advice有点不同的是,在定义Poincut表示式时,多了一个returning属性,用于指定目标方法执行完后的返回值。
XML文件中的设置与LogBeforeAdvice的相似(将logBeforeAdvice的定义改为logAfterReturning的定义),不再列举。
五、Around Advice:基于XML Sechma
在Spring 2.0中,Around Advice不用实现org.aoplliance.intercept.MethodInterceptor接口,但Advice的方法必须返回对象,并且必须定义一个ProceedingJoinPoint参数,例如:
java 代码
package com.savage.aop; import org.aspectj.lang.ProceedingJoinPoint; public class LogAroundAdvice { public Object invoke(ProceedingJoinPoint joinPoint) { System.out.println("Logging before " + joinPoint.getSignature().getName()); Object retVal = joinPoint.proceed(); System.out.println("Logging after " + joinPoint.getSignature().getName()); return retVal; } }
XML中的设置如下:
xml 代码
<?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.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd"> <bean id="messageSender" class="com.savage.aop.HttpMessageSender"></bean> <bean id="logAroundAdvice" class="com.savage.aop.LogAroundAdvice"></bean> <aop:config> <aop:aspect id="logAround" ref="logAroundAdvice"> <aop:around pointcut="execution(* com.savage.aop.MessageSender.*(..))" method="invoke"/> </aop:aspect> </aop:config> </beans>
六、Around Advice:基于Annotation
和Before Advice相似,使用@Around来表示Around Advice:
java 代码
package com.liuyoung.aop; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; @Aspect public class LogAroundAdvice { @Around("execution(* com.savage.aop.MessageSender.*(..))") public Object invoke(ProceedingJoinPoint joinPoint) throws Throwable { System.out.println("Logging before [log around advice] : " + joinPoint.getSignature().getName()); Object retVal = joinPoint.proceed(); System.out.println("Logging after [log around advice] : " + joinPoint.getSignature().getName()); return retVal; } }
七、Throw Advice:基于XML Sechma
在Spring 2.0中,Throw Advice不用实现org.springframework.aop.ThrowsAdvice接口,但Advice的方法必须定义Throwable(或其子类)参数,例如:
java 代码
package com.savage.aop; import org.aspectj.lang.JoinPoint; public class LogThrowingAdvice { public void afterThrowing (JoinPoint joinPoint, Throwable throwable) { System.out.println("Logging when throwing " + joinPoint.getSignature().getName()); } }
在XML的设置如下:
xml 代码
<?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.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd"> <bean id="messageSender" class="com.savage.aop.HttpMessageSender"></bean> <bean id="logThrowingAdvice" class="com.savage.aop.LogThrowingAdvice"></bean> <aop:config> <aop:aspect id="logThrowing" ref="logThrowingAdvice"> <aop:after-throwing pointcut="execution(* com.savage.aop.MessageSender.*(..))" throwing="throwable" method="afterThrowing"/> </aop:aspect> </aop:config> </beans>
在<aop:after-throwing></aop:after-throwing>中必须定义throwing属性,指定方法中的throwable参数。Spring将根据异常类型决定是否调用afterThrowing方法。
八、Throw Advice:基于Annotation
java 代码
package com.savage.aop; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.AfterThrowing; @Aspect public class AfterThrowingAdvice { @AfterThrowing(pointcut="execution(* com.savage.aop.MessageSender.*(..))", throwing="throwable") public void afterThrowing(JoinPoint joinPoint, Throwable throwable) { System.out.println("Logging when throwing " + joinPoint.getSignature().getName()); } }
相关文章推荐
- spring2.0基于Annotation的AOP实现以及AOP事务
- Spring 2.0中的AOP实现(转)
- aop实现-spring2.0中的aop实现
- AOP实现(三)——Spring 2.0中Pointcut的定义
- AOP实现——Spring 2.0中Pointcut的定义
- Spring2.0 AOP 的实现
- AOP实现(三)--spring2.0中pointcut定义
- spring2.0基于XML Schema的AOP实现以及AOP事务
- Spring_Spring_教程12_Spring利用注解实现Aop
- spring2.0 配置AOP(切面)
- 8 -- 深入使用Spring -- 4...2 使用AspectJ实现AOP
- Spring_AOP_Proxy AOP动态代理的实现
- 使用Spring的注解方式实现AOP
- Spring实现AOP的4种方式
- Spring Aop自定义注解实现日志管理
- 关于spring 2.0自定义xml 标记 (二 如何实现)
- Spring实现AOP的多种方式
- spring 实现AOP所需的三个jar包
- 使用Spring的注解方式实现AOP的细节