1、@Aspect
- 配置切面Bean,和<bean.../>元素进行配置无区别,一样支持依赖注入来配置属性值;
- 如果启动了Spring的“零配置”特性,一样可以让Spring自动搜索,并加载指定路径下的切面Bean
- 使用@Aspect修饰的类(切面类)和其他类一样可以有方法、成员变量定义,还可能包括切入点、增强处理定义
- 使用@Aspect修饰类,Spring将不会把该Bean当成组件Bean处理,因此负责自动增强的后处理Bean将会略过该Bean,不会对该Bean进行任何增强处理
2、@Before
- 修饰一个方法时,该方法将作为Before增强处理
- 使用@Before修饰事,需要指定一个value属性值,该属性值指定一个切入点表达式(既可以是一个已有的切入点,也可以直接定义切入点表达式),用于指定该增强处理将被织入哪些切入点
- 表示在切入点执行前需要进行的操作或者需要执行的方法
3、@After
- 同Before
- 表示在切入点执行后,进行哪些操作
- 通常用于资源释放
4、@AfterReturning
- 同上
- 使用@AfterReturning增强处理可指定如下常用属性
-
- pointcut/value:用于指定该切入点对应的切入表达式,当指定了pointcut属性值后,value属性值将会被覆盖
-
- returning:该属性值指定一个形参名,用于表示Advuce方法中可以定义与此同名的形参,该形参可用于访问目标方法的返回值
-
-
- 在Advice方法中定义该形参时指定的类型,会限制目标方法必须返回指定类型的值或没有返回值
- AfterReturning增强处理只有在目标方法成功完成后才会被织入
- After增强处理不管目标方法时如何结束的(成功或抛出异常),它都会被织入
5、@AfterThrowing
- 同上
- 使用@AfterThrowing注解时可指定如下两个常用属性:
-
- pointcut/value:用于指定该切入点对应的切入表达式,当指定了pointcut属性值后,value属性值将会被覆盖
-
- throwing:该属性值指定一个形参名,用于表示Advice方法中可以定义与此同名的形参,该形参可用于访问目标方法抛出的异常
-
- 在Advice方法中定义该形参时指定的类型,会限制目标方法必须抛出指定类型的异常
6、@Around
- Around增强处理是功能比较强大的增强处理;
- 近似等于Before增强处理和AfterReturning增强处理的总和
- 既可以在执行目标方法之前织入增强动作,也可以在执行目标方法之后织入增强动作
- Around增强处理可以改变执行目标方法的参数值,也可以改变执行目标方法之后的返回值
- Around增强处理虽然功能强大,但通常需要在线程安全的环境下使用,所以一般用Before和AfterReturning增强处理能解决的问题,不建议用Around
- 如果需要目标方法执行之前和之后共享某种状态数据,则应该考虑使用Around增强处理;尤其是需要改变目标方法的返回值时,则只能使用Around增强处理了
- @Around增强处理事,需要指定一个value属性,该属性指定该增强处理被植入的切入点
- 当定义一个Around增强处理方法时,该方法的第一个形参必须是ProceedingJoinPoint类型(至少包含一个形参),在增强处理方法体内,调用ProceedingJoinPoint参数的proceed()方法才会执行目标方法——这就是Around增强处理可以完全控制目标方法的执行时机、如何执行的关键;如果程序没有调用ProceedingJoinPoint参数的proceed()方法,则目标方法不会被执行。
- 调用ProceedingJoinPoint参数的proceed()方法时,还可以传入一个Object[]对象作为参数,该数组中的值将被传入目标方法作为执行方法的实参
package com.sysker.aspect;
import java.util.Arrays;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
@Aspect
public class TxAspect {
@Around("execution(* com.sysker.impl.*.*(..))")
public Object changeValue(ProceedingJoinPoint jp) throws Throwable {
System.out.println("TxAspect执行目标方法之前,模拟开始事务。。。");
Object[] args = jp.getArgs();
if(args != null && args.length > 1) {
args[0] = "change00" + args[0];
}
Object rvt = jp.proceed(args);
System.out.println("TxAspect执行目标方法之后,模拟结束事务。。。");
if(rvt != null && rvt instanceof Integer) {
rvt = (Integer)rvt*(Integer)rvt;
}
System.out.println(rvt);
return rvt;
}
@Before("execution(* com.sysker.impl.*.*(..))")
public void authority(JoinPoint jp) {
System.out.println("TxAspect模拟执行权限检查");
System.out.println("Before增强:被织入增强处理的目标方法:" + jp.getSignature().getName());
System.out.println("Before增强:被织入增强处理的目标方法的参数:" + Arrays.toString(jp.getArgs()));
System.out.println("Before增强:被织入增强处理的目标对象为:" + jp.getTarget());
}
@After("execution(* com.sysker.impl.*.*(..))")
public void show() {
System.out.println("TxAspect模拟释放资源");
}
@AfterThrowing(throwing = "ex", pointcut = "execution(* com.sysker.impl.*.*(..))")
public void doRecoveryAciton(Throwable ex) {
System.out.println("目标抛出的异常:" + ex);
System.out.println("模拟Advice对异常进行修复");
}
}