SpringBoot项⽬中使⽤AOP的⽅法
本⽂介绍了SpringBoot项⽬中使⽤AOP的⽅法,分享给⼤家,具体如下:
1.概述
将通⽤的逻辑⽤AOP技术实现可以极⼤的简化程序的编写,例如验签、鉴权等。Spring的声明式事务也是通过AOP技术实现的。
Spring的AOP技术主要有4个核⼼概念:
Pointcut: 切点,⽤于定义哪个⽅法会被拦截,例如 execution(* cn.springcamp.springaop.service.*.*(..))
Advice: 拦截到⽅法后要执⾏的动作
Aspect: 切⾯,把Pointcut和Advice组合在⼀起形成⼀个切⾯
Join Point: 在执⾏时Pointcut的⼀个实例
Weaver: 实现AOP的框架,例如 AspectJ 或 Spring AOP
2. 切点定义
常⽤的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类。
3. 常⽤的切⾯
Before: 在⽅法执⾏之前执⾏Advice,常⽤于验签、鉴权等。
After: 在⽅法执⾏完成后执⾏,⽆论是执⾏成功还是抛出异常.
AfterReturning: 仅在⽅法执⾏成功后执⾏.
AfterThrowing: 仅在⽅法执抛出异常后执⾏.
⼀个简单的Aspect:
@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);
}
}
4. ⾃定义注解
假设我们想收集特定⽅法的执⾏时间,⼀种⽐较合理的⽅式是⾃定义⼀个注解,然后在需要收集执⾏时间的⽅法上加上这个注解。
⾸先定义⼀个注解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)
5.总结
在运⾏⽰例项⽬时,控制台会输出以下内容:
-------------> 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
foospring aop应用场景
可以看出⼏种 Aspect 的执⾏顺序依次为 Before After Around AfterReturning(AfterThrowing)
以上就是本⽂的全部内容,希望对⼤家的学习有所帮助,也希望⼤家多多⽀持。

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。