面向切面编程基础定义
多个对象需要重用的通用功能,我们常想到的对象技术有继承和聚合。但是继承往往会导致一个脆弱的对象体系,而聚合需要对聚合对象进行显示复杂的调用。
切面编程提供了取代它们的另一种方案。我们仍然在一个地方定义通用功能,但是不需要像聚合那样在使用位置显示调用,我们通过声明的方式定义这个功能以何种方式在何处应用,不需要修改受影响的类,至于如何声明我们接下来会讲。
我们将通用功能写在一个类中,这些类就叫做切面。
切面是通知和切点的组合。
-
通知:定义了切面要完成的工作以及何时使用。之前?之后?之前和之后都调用?还是只在方法抛出异常时调用?
-
连接点:通知被运用的某个具体位置。
-
切点:定义了通知在何处使用,在程序中它相当于某一类连接点的通配表达式。
spring定义了五种通知:
- 前置通知:目标方法调用之前调用通知功能。 注解:@Before
- 后置通知:目标方法调用之后调用通知功能。 注解:@After
- 返回通知:目标方法调用成功执行后调用通知功能。 注解:@AfterReturning
- 异常通知:目标方法调用抛出异常后调用通知功能。 注解:@AfterThrowing
- 环绕通知:拦截目标方法的调用,通知方法(ProceedingJoinPoint jp)入参会被自动注入,在通知中自定义何处何时通过jp.proceed()调用目标方法,即完全自定义。 注解:@Around
连接点:前面所说的何时,何处便是相对于此。也就是切面被应用的位置。
- spring aop是基于动态代理的,所以只支持方法级别的连接点
AspectJ除了方法切点,它们还提供了字段和构造器接入点。所以spring aop无法拦截对象字段的修改,无法再bean创建时应用通知。
Spring提供了4种类型的AOP支持(声明方式):
- 基于代理的经典Spring AOP; -----------------------即是动态代理。相对于其他方法太过笨重,已经被淘汰。
- 纯POJO切面;--------------------------------------------xml配置
- @AspectJ注解驱动的切面;--------------------------注解
- 注入式AspectJ切面(适用于Spring各版本)------AOP需求超过了简单的方法调用(如构造器或属性拦截),使用Aspectj来使用切面。
编码
通过切点选择连接点
注意只有execution指示器是实际执行匹配的,而其他的指示器都是用来限制匹配的。
一个完整的切面声明代码:
//@Aspect切面声明、@Component必须是一个POJO(spring 组件)
@Component
@Aspect
public class Audience {
//@Pointcut注解能够在一个@AspectJ切面内定义可重用的切点,execution表达式指明了切点的位置
@Pointcut(value = "execution(* com.zhu.springapp.pojo.Performance.perform(..)) && bean('performance')")
public void pointCut(){};
//为切点添加前置通知
@Before("pointCut()")
//前置通知
public void before(){
System.out.println("before()...");
}
//为切点添加环绕通知
@Around("pointCut()")
public void around(ProceedingJoinPoint joinPoint){
try{
System.out.println("before()...");
joinPoint.proceed();
System.out.println("afterReturning()...");
}catch (Throwable throwable) {
System.out.println("AfterThrowing()...");
throwable.printStackTrace();
}finally{
System.out.println("after()...");
}
}
}
为通知传递参数
使用切点表达式中的args()限定符,它表明传递给连接点的的参数也会传递到通知中去。
//args(aa)的参数名aa要和pointCut2(int aa)参数名aa一致
@Pointcut(value = "execution(* com.zhu.springapp.pojo.Teee.hh(int)) " + "&& args(aa)")
public void pointCut2(int aa){};
//这里也要一致
@Before("pointCut2(bb)")
public void hhBefore(int bb){
System.out.println(bb);
}