专栏首页以Java架构赢天下面试:Spring Boot项目怎么用AOP

面试:Spring Boot项目怎么用AOP

1.概述

将通用的逻辑用AOP技术实现可以极大的简化程序的编写,例如验签、鉴权等。Spring的声明式事务也是通过AOP技术实现的。 Spring的AOP技术主要有4个核心概念:

  1. Pointcut: 切点,用于定义哪个方法会被拦截,例如 execution(* cn.springcamp.springaop.service.*.*(..))
  2. Advice: 拦截到方法后要执行的动作
  3. Aspect: 切面,把PointcutAdvice组合在一起形成一个切面
  4. Join Point: 在执行时Pointcut的一个实例
  5. Weaver: 实现AOP的框架,例如 AspectJSpring 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.常用的切面

  1. Before: 在方法执行之执行Advice,常用于验签、鉴权等
  2. After: 在方法执行完成执行,无论是执行成功还是抛出异常.
  3. AfterReturning: 仅在方法执行成功后执行.
  4. 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 foo

可以看出几种 Aspect 的执行顺序依次为 Before After Around AfterReturning(AfterThrowing)

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 设置事务超时时间的问题及Oracle数据库update和锁

    写在前面:2020年面试必备的Java后端进阶面试题总结了一份复习指南在Github上,内容详细,图文并茂,有需要学习的朋友可以Star一下! GitHub地...

    用户5546570
  • 写时复制技术(详解版)

    我们知道了一个进程如何采用请求调页,仅调入包括第一条指令的页面,从而能够很 快开始执行。然而,通过系统调用 fork() 的进程创建最初可以通过使用类似于页面共...

    用户5546570
  • Spring Boot最常用的25个注解,干货了解一下

    用户5546570
  • 学习 sentry 源码整体架构,打造属于自己的前端异常监控SDK

    这是学习源码整体架构第四篇。整体架构这词语好像有点大,姑且就算是源码整体结构吧,主要就是学习是代码整体结构,不深究其他不是主线的具体函数的实现。文章学习的是打包...

    若川
  • 自媒体创作常用工具推荐

    这是视频的链接地址:https://www.bilibili.com/video/BV1nJ411P7Xp

    萌萌哒的瓤瓤
  • 自媒体创作常用工具推荐

    这是视频的链接地址:https://www.bilibili.com/video/BV1nJ411P7Xp

    萌萌哒的瓤瓤
  • DevOps工具链在公司中扮演的关键角色

    DevOps工具链是一组用于执行复杂软件交付任务的数字工具。工具链中的工具通常一个接一个地执行,其中一个工具的输出是下一个工具的输入。

    陈琦聊测试
  • 第200天:js---常用string原型扩展

    半指温柔乐
  • golang context实战

    来自官方文档: https://blog.golang.org/context: Incoming requests to a server should cr...

    王磊-AI基础
  • 装饰者模式、观察者模式、拦截器模式及其Go语言实现

    (1)什么是装饰者模式? 装饰者模式是在不使用继承和不改变原类文件的情况下,动态地扩展一个对象的功能。在设计模式的八大原则中,开闭原则规定了对扩展开放,对修改...

    魏晓蕾

扫码关注云+社区

领取腾讯云代金券