spring定义一个切面是件麻烦的事情,需要实现专门的接口,还要进行一些较为复杂的配置,有没有较为简单的方法???
@AspectJ注解可以很容易定义一个切面,且不需要实现任何的接口。缺点是对JDK的版本有限制,要求是5.0以上
当然对于不足5.0的版本,可以通过Schema的配置定义切面,方便程度和@AspectJ相差无几。
无论是基于XML配置的AOP还是基于@AspectJ注解的AOP,只是在表达方式有所不同,底层都是采用动态代理技术(JDK代理或CGLib代理)。
注解是代码的附属信息,它遵循一个原则:注解不能直接干扰程序代码的运行,无论是增加或删除注解,代码都能正常运行。java语言解释器会忽略这些注解,而由第三方工具对注解进行解析,从而达到间接控制程序代码的运行,它们通过java反射机制读取注解的信息。
Pointcut和Advice分别表切点和增强,并用Advisor将两者整体为切面,@AspectJ则是采用注解的方式来描述切点、增强,两者只是表述上有所不同,本质是一样的。
代码示例:
切面
import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; @Aspect public class PreGreetingAspect{ @Before("execution(* greetTo(..))") public void beforeGreeting(){ System.out.println("How are you"); } }a. PreGreetingAspect类定义处,标注了@Aspect注解,表示其为一个切面。
b. beforeGreeting方法上,Before表示是前置增强;execution(* greetTo(..))表示一个切点表达式,在目标类的greetTo()方法上织入增强,greetTo方法可以带任意入参和任意返回值。
c. beforeGreeting方法体,表示具体的横切逻辑。
PreGreetingAspect类通过注解和代码,将切点、增强类型、增强逻辑糅合到一个类中。
配置文件
<aop:aspectj-autoproxy/> //目标bean <bean id="waiter" class="com.baobaotao.NaiveWaiter" /> //采用@AspectJ注解的切面类 <bean class="com.aspectj.example.PreGreetingAspect" /> //自动代理创建器,自动将@AspectJ切面类织入到目标Bean中 <bean class="org.springframework.aop.aspectj.annotation.AnnotationAwareAspectJAutoProxyCreator"/>
execution为关键字(ps:表示目标类执行某一方法)
* greetTo(..)为操作参数(ps:表示目标方法的匹配模式串)
目前spring支持9个@AspectJ切点表达式函数,他们用不同的方式描述目标类的连接点,大致分为4种类型:
1)方法切点函数:通过描述目标类方法信息定义连接点,如execution
2)方法入参切点函数:通过描述目标类方法入参的信息定义连接点
3)目标类切点函数:通过描述目标类类型信息定义连接点
4)代理类切点函数:通过描述目标类的代理类的信息定义连接点
增强类型:
@Before 前置增强
@AfterReturning 后置增强
@Around 环绕增强
@AfterThrowing 抛出异常的增强
@DeclareParents 引介增强
与xml形式的基本一致,只是换了种表达方式,采用了注解的形式来描述。
---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
对于不满足JDK5.0的项目,只能望@AspectJ而兴叹了。没关系,我们还可以用Schema配置切面
说白了,其实就是将注解信息移到了Schema的XML配置文件中,形式不同罢了。
简单实例:
xml配置文件
<aop:config proxy-target-class="true"> //增强方法所在的bean <aop:aspect ref="adviceMethods"> //前置增强 <aop:before method="preGreeting" //切点表达式,与@AspectJ的语法相同 pointcut="target(com.baobaotao.NaiveWaiter) and args(name)" arg-names="name" /> //后置增强 <aop:after-returning method="afterReturning" pointcut="target(com.baobaotao.SmartSeller)" returning="retVal" /> </aop:aspect> </aop:config> <bean id="adviceMethods" class="com.baobaotao.schema.AdviceMethods" /> <bean id="naiveWaiter" class="com.baobaotao.NaiveWaiter" /> <bean id="naughtyWaiter" class="com.baobaotao.NaughtyWaiter" /> <bean id="seller" class="com.baobaotao.SmartSeller" />
其中
1、 <aop:aspect>元素定义切面,内部可以定义多个增强
2、 <aop:config>拥有一个proxy-target-class属性,true时表示声明的切面采用CGLib动态代理技术;反之表示使用JDK动态代理技术
public class AdviceMethods { public void preGreeting(String name) { System.out.println("--how are you!--"); System.out.println(name); } //后置增强对应方法 public void afterReturning(int retVal){ System.out.println("----afterReturning()----"); System.out.println("returnValue:"+retVal); System.out.println("----afterReturning()----"); } }