具体实践

Spring AOP是Spring框架中一个支持实现面向切面编程的模块,由于Spring Boot已经把Spring框架组合得非常好用,所以在基于Spring Boot框架的项目中实现AOP编程也是非常方便,具体来说可以分为如下几步:

第一步: 在项目中引入依赖配置。

<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>

第二步: 定义切面,切入点和通知,三者可以直接在一个类中完成。

使用注解@Aspect来定义切面(注:需要在应用注解@Aspect的类上使用注解@Component标识这是一个容器管理的Bean),使用注解@Pointcut来定义切入点,使用注解@Before@After@Around@AfterReturning@AfterThrowing来定义不同的通知。

@Aspect // 使用注解@Aspect定义切面
@Component // 同时使用注解@Component标识这是一个IoC容器管理的Bean
public class AspectSample {
// 定义切点
// 使用@Pointcut注解定义切点
// 切点的表达式有多种,如:execution,within,@annotation,bean()
// 详见:https://docs.spring.io/spring-framework/reference/core/aop/ataspectj/pointcuts.html#aop-pointcuts-combining // 匹配类org.chench.springboot.scaffolding.controller.AOPController的所有方法
@Pointcut("execution(* org.chench.springboot.scaffolding.controller.AOPController.*(..))")
public void pointcutSample() {} // 定义通知
// 可以用于定义通知的注解有5个:
// 1. @Before(前置通知:目标方法执行前执行)
// 2. @After(后知通知:无论目标方法正常返回还是抛出异常都执行)
// 3. @Around(环绕通知:目标方法执行前和返回后执行)
// 4. @AfterReturning(返回通知:目标方法正常返回后执行)
// 5. @AfterThrowing(异常通知:目标方法抛出异常后执行) // 前置通知:目标方法执行前执行
@Before("pointcutSample()")
public void beforeAdvice(JoinPoint joinPoint) {
System.out.println(String.format("这里是前置通知执行:%s", new Date()));
} // 后置通知:无论目标方法正常返回还是抛出异常都执行
@After("pointcutSample()")
public void afterAdvice(JoinPoint joinPoint) {
System.out.println(String.format("这里是后置通知执行:%s", new Date()));
} // 环绕通知执行:目标方法执行前和返回后执行
// 注意:
// 1.在环绕通知中一定要返回目标方法的返回值,否则客户端就无法接收到结果啦
// 2.如果在环绕通知中捕获了目标方法执行时抛出的异常,则异常通知对应的切面逻辑将得不到执行
@Around("pointcutSample()")
public Object aroundAdvice(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
System.out.println(String.format("这里是环绕通知执行前:%s", new Date()));
Object result = proceedingJoinPoint.proceed();
System.out.println(String.format("这里是环绕通知执行后:%s", new Date()));
return result;
} // 返回通知:目标方法正常返回后执行
@AfterReturning(value = "pointcutSample()", returning = "r")
public void afterRunningAdvice(JoinPoint joinPoint, Object r) {
System.out.println(String.format("这里是返回通知执行:%s %s", new Date(), r));
} // 异常通知:目标方法抛出异常后执行
// 有意思的是:
// 如果在项目中同时使用了@ControllerAdvice注解进行全局异常处理,同时也定义了异常通知
// 那么异常通知将在@ControllerAdvice之前得到执行
@AfterThrowing(value = "pointcutSample()", throwing = "e")
public void afterThrowingAdvice(JoinPoint joinPoint, Throwable e) {
System.out.println(String.format("这里是异常通知执行:%s", new Date()));
}
}

写在最后

关于Spring对AOP的支持有如下几点总结:

1.切入点只支持方法

2.不同类型的通知执行的先后顺序不同,假设对于一个相同的切入点定义了所有通知类型。

  • 当目标方法没有抛出异常时,各个通知的执行顺序如下:
这里是aroundAdvice中目标方法执行前:Tue Jul 18 00:20:31 CST 2023
这里是beforeAdvice执行:Tue Jul 18 00:20:31 CST 2023
执行目标方法m1:Tue Jul 18 00:20:31 CST 2023
这里是aroundAdvice中目标方法执行后:Tue Jul 18 00:20:31 CST 2023
这里是afterAdvice执行:Tue Jul 18 00:20:31 CST 2023
这里是afterRunningAdvice执行:Tue Jul 18 00:20:31 CST 2023
  • 当目标方法执行时抛出异常,且在环绕通知中没有明确捕获该异常,则各个通知的执行顺序如下:
这里是aroundAdvice中目标方法执行前:Tue Jul 18 00:22:54 CST 2023
这里是beforeAdvice执行:Tue Jul 18 00:22:54 CST 2023
执行目标方法m2:Tue Jul 18 00:22:54 CST 2023
这里是afterAdvice执行:Tue Jul 18 00:22:54 CST 2023
这里是afterThrowingAdvice执行:Tue Jul 18 00:22:54 CST 2023

显然,此时就缺少了这里是aroundAdvice中目标方法执行后这里是afterRunningAdvice执行,但是这里是afterThrowingAdvice执行却出现了。

3.如果使用了@ControllerAdvice实现全局异常拦截,同时也定义了异常通知@AfterThrowing,那么异常通知@AfterThrowing将在@ControllerAdvice之前得到执行。

@ControllerAdvice
public class GlobalExceptionAdvice {
@ResponseBody
@ExceptionHandler(value = Exception.class)
public Object handleException(HttpServletRequest req, HttpServletResponse resp, Exception ex) {
System.out.println(String.format("这里是@ControllerAdvice:%s", new Date()));
return JsonVO.build().setCode(-1).setMessage("error");
}
}

当目标方法抛出异常后,则输出:

这里是aroundAdvice中目标方法执行前:Tue Jul 18 00:28:48 CST 2023
这里是beforeAdvice执行:Tue Jul 18 00:28:48 CST 2023
执行目标方法m2:Tue Jul 18 00:28:48 CST 2023
这里是afterAdvice执行:Tue Jul 18 00:28:48 CST 2023
这里是afterThrowingAdvice执行:Tue Jul 18 00:28:48 CST 2023 # @AfterThrowing将在@ControllerAdvice之前得到执行
这里是@ControllerAdvice:Tue Jul 18 00:28:48 CST 2023

4.每种通知方法中都可以通过第一个参数获取JoinPoint对象(环绕通知的方法参数ProceedingJoinPoint也是JoinPoint类型),并且在@AfterReturning通知中还可以获取目标方法的返回值,在@AfterThrowing通知中可以获取目标方法抛出的异常。

5.拦截器HandlerInterceptor也是Spring AOP的一个具体应用,当同时存在拦截器HandlerInterceptor和通知Advice时,拦截器会在通知之前执行,因为拦截器处理的是uri,而通知处理的是方法,一个在前,一个在后。Spring提供的拦截器机制跟Servlet中的Filter机制很像,但是他们属于不同范畴(后者属于Servlet规范中的内容,而前者属于Spring框架提供的支持)。

【参考】

aop-pointcuts-combining

Spring AOP - 注解方式使用介绍

原来这才是Spring Boot使用AOP的正确姿势

在SpringBoot中使用AOP——通知中的参数

AOP通知获取数据(参数、返回值、异常)

Spring AOP使用:自定义注解、通知(简单使用和原理了解)

SpringBoot之Filter注册

优雅的使用SpringBoot 中的Filter

Spring Boot拦截器(Interceptor)详解

Spring 过滤器 拦截器 AOP区别

在SpringBoot中实践AOP编程的更多相关文章

  1. Springboot中使用AOP统一处理Web请求日志

    title: Springboot中使用AOP统一处理Web请求日志 date: 2017-04-26 16:30:48 tags: ['Spring Boot','AOP'] categories: ...

  2. SpringBoot图文教程5—SpringBoot 中使用Aop

    有天上飞的概念,就要有落地的实现 概念+代码实现是本文的特点,教程将涵盖完整的图文教程,代码案例 文章结尾配套自测面试题,学完技术自我测试更扎实 概念十遍不如代码一遍,朋友,希望你把文中所有的代码案例 ...

  3. 在SpringBoot中配置aop

    前言 aop作为spring的一个强大的功能经常被使用,aop的应用场景有很多,但是实际的应用还是需要根据实际的业务来进行实现.这里就以打印日志作为例子,在SpringBoot中配置aop 已经加入我 ...

  4. 编写SpringBoot 中的AOP

    编写SpringBoot 中的AOP 在程序开发的过程中会使用到AOP的思想,面向切面进行开发,比如登录的验证,记录日志等等-频繁需要操作的步骤,在遇到这种情况时就要使用Spring 的AOP了 Sp ...

  5. 聊Javascript中的AOP编程

    Duck punch 我们先不谈AOP编程,先从duck punch编程谈起. 如果你去wikipedia中查找duck punch,你查阅到的应该是monkey patch这个词条.根据解释,Mon ...

  6. 聊聊Javascript中的AOP编程

    Duck punch 我们先不谈AOP编程,先从duck punch编程谈起. 如果你去wikipedia中查找duck punch,你查阅到的应该是monkey patch这个词条.根据解释,Mon ...

  7. spring-boot中的AOP

    public class User { private Integer id; private String username; private String note; public User(In ...

  8. SpringBoot中搭配AOP实现自定义注解

    1 springBoot的依赖 确定项目中包含可以注解的依赖 <dependency> <groupId>org.springframework.boot</groupI ...

  9. .NET Core中实现AOP编程

    AOP全称Aspect Oriented Progarmming(面向切面编程),其实AOP对ASP.NET程序员来说一点都不神秘,你也许早就通过Filter来完成一些通用的功能,例如你使用Autho ...

  10. (转).NET Core中实现AOP编程

    原文地址:https://www.cnblogs.com/xiandnc/p/10088159.html AOP全称Aspect Oriented Progarmming(面向切面编程),其实AOP对 ...

随机推荐

  1. [转帖]Percolator - 分布式事务的理解与分析

    https://zhuanlan.zhihu.com/p/261115166 Percolator - 分布式事务的理解与分析 概述 一个web页面能不能被Google搜索到,取决于它是否被Googl ...

  2. [转帖]Jmeter压力测试工具安装及使用教程

    https://www.cnblogs.com/monjeo/p/9330464.html 一.Jmeter下载 进入官网:http://jmeter.apache.org/ 1.第一步进入官网如下图 ...

  3. [转帖]深入理解mysql-第十章 mysql查询优化-Explain 详解(上)

    目录 一.初识Explain 二.执行计划-table属性 三.执行计划-id属性 四.执行计划-select_type属性 一条查询语句在经过MySQL查询优化器的各种基于成本和规则的优化会后生成一 ...

  4. 【转帖】网卡bonding模式 - bond0、1、4配置

    网卡bonding简介 网卡绑定就是把多张物理网卡通过软件虚拟成一个虚拟的网卡,配置完毕后,所有的物理网卡的ip和mac将会变成相同的.多网卡同时工作可以提高网络速度,还可以实现网卡的负载均衡.冗余. ...

  5. [转帖]RFC1180

    [译] RFC 1180:朴素 TCP/IP 教程(1991) 译者序 本文翻译自 1991 年的一份 RFC(1180): A TCP/IP Tutorial. 本文虽距今将近 20 年,但内容并未 ...

  6. 二进制安装Mysql数据库的快速方法

    二进制安装Mysql数据库的快速方法 摘要 还是国产操作系统 rpm包可能不太兼容,为了简单准备使用tar包方式安装mysql数据库 这里简单记录一下过程. 为以后使用. 介质下载 下载二进制的tar ...

  7. Booking.com如何在毫秒内搜索数百万个地点

    译自:How Booking.com Searches Through Millions of Locations in Milliseconds Booking.com是一家与酒店.旅馆.度假租赁等 ...

  8. PHP GC回收机制详解

    前言 GC的全称是Garbage Collection也就是垃圾回收的意思,在PHP中,是使用引用计数和回收周期来自动管理内存对象的,当一个对象被设置为NULL,或者没有任何指针指向时,他就会变成垃圾 ...

  9. 【小测试】golang中数组边界检查的开销大约是1.87%~3.12%

    作者:张富春(ahfuzhang),转载时请注明作者和引用链接,谢谢! cnblogs博客 zhihu Github 公众号:一本正经的瞎扯 对比C/C++, golang等类型安全的语言会在数组访问 ...

  10. 自动化部署实例(donetcore GitLab CICD )

    主要简单的介绍了一下 GitLab CI 的持续集成以及持续部署,这篇将通过 GitLab CI 发布一个 .net core 项目,来带小伙伴们感受一下自动化的魅力,从此告别手动发布. 准备工作 创 ...