将通用的逻辑用AOP技术实现可以极大的简化程序的编写,例如验签、鉴权
等。Spring的声明式事务
也是通过AOP技术实现的。
Spring的AOP技术主要有4个核心概念:
execution(* cn.springcamp.springaop.service.*.*(..))
Pointcut
和Advice
组合在一起形成一个切面Pointcut
的一个实例AspectJ
或 Spring AOP
常用的Pointcut定义有 execution
和 @annotation
两种。execution
定义对方法无侵入,用于实现比较通用的切面。@annotation
可以作为注解加到特定的方法上,例如Spring的Transaction
注解。
execution切点定义应该放在一个公共的类中,集中管理切点定义。
示例:
public class CommonJoinPointConfig { @Pointcut("execution(* cn.springcamp.springaop.service.*.*(..))") public void serviceLayerExecution() {} } 复制代码这样在具体的Aspect类中可以通过 CommonJoinPointConfig.serviceLayerExecution()来引用切点。 public class BeforeAspect { @Before("CommonJoinPointConfig.serviceLayerExecution()") public void before(JoinPoint joinPoint) { System.out.println(" -------------> Before Aspect "); System.out.println(" -------------> before execution of " + joinPoint); } }
当切点需要改变时,只需修改CommonJoinPointConfig
类即可,不用修改每个Aspect
类。
前
执行Advice,常用于验签、鉴权等
。后
执行,无论是执行成功还是抛出异常
.成功后
执行.异常后
执行.@Aspect @Component public class BeforeAspect { @Before("CommonJoinPointConfig.serviceLayerExecution()") public void before(JoinPoint joinPoint) { System.out.println(" -------------> Before Aspect "); System.out.println(" -------------> before execution of " + joinPoint); } }
假设我们想收集特定方法的执行时间,一种比较合理的方式是自定义一个注解,然后在需要收集执行时间的方法上加上这个注解。
TrackTime
:@Target({ElementType.METHOD, ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) public @interface TrackTime { String param() default ""; }
Aspect
类,用于实现注解的行为:@Aspect @Component public class TrackTimeAspect { @Around("@annotation(trackTime)") public Object around(ProceedingJoinPoint joinPoint, TrackTime trackTime) throws Throwable { Object result = null; long startTime = System.currentTimeMillis(); result = joinPoint.proceed(); long timeTaken = System.currentTimeMillis() - startTime; System.out.println(" -------------> Time Taken by " + joinPoint + " with param[" + trackTime.param() + "] is " + timeTaken); return result; } }
在某个方法上使用这个注解,就可以收集这个方法的执行时间:
@TrackTime(param = "myService") public String runFoo() { System.out.println(" -------------> foo"); return "foo"; }
注意@TrackTime(param = "myService")
注解是可以传参的
为了让注解可以传参数,需要在定义注解时指定一个参数String param() default
"默认值",
同时在Aspect类中,around方法上加上相应的参数,@Around
注解中也需要用参数的变量名trackTime
,而不能用类名TrackTime
。
@Around("@annotation(trackTime)") public Object around(ProceedingJoinPoint joinPoint, TrackTime trackTime)
在运行示例项目时,控制台会输出以下内容:
-------------> Before Aspect -------------> before execution of execution(String cn.springcamp.springaop.service.MyService.runFoo()) -------------> foo -------------> Time Taken by execution(String cn.springcamp.springaop.service.MyService.runFoo()) with param[myService] is 8 -------------> After Aspect -------------> after execution of execution(String cn.springcamp.springaop.service.MyService.runFoo()) -------------> AfterReturning Aspect -------------> execution(String cn.springcamp.springaop.service.MyService.runFoo()) returned with value foo
可以看出几种 Aspect 的执行顺序依次为 Before After Around AfterReturning(AfterThrowing)
本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。
我来说两句