Spring Boot使用AOP

一、为什么需要面向切面编程?

面向对象编程(OOP)的好处是显而易见的,缺点也同样明显。当需要为多个不具有继承关系的对象添加一个公共的方法的时候,例如日志记录、性能监控等,如果采用面向对象编程的方法,需要在每个对象里面都添加相同的方法,这样就产生了较大的重复工作量和大量的重复代码,不利于维护。面向切面编程(AOP)是面向对象编程的补充,简单来说就是统一处理某一"切面"的问题的编程思想。如果使用AOP的方式进行日志的记录和处理,所有的日志代码都集中于一处,不需要再每个方法里面都去添加,极大减少了重复代码。

二、Spring AOP术语

通知(Advice)包含了需要用于多个应用对象的横切行为,就是定义了"什么时候"和"做什么"。

连接点(Join Point)是程序执行过程中能够应用通知的所有点。

切点(Poincut)是定义了在"什么地方"进行切入,哪些连接点会得到通知。显然,切点一定是连接点。

切面(Aspect)是通知和切点的结合。通知和切点共同定义了切面的全部内容------是什么,何时,何地完成功能。

引入(Introduction)允许我们向现有的类中添加新方法或者属性。

织入(Weaving)是把切面应用到目标对象并创建新的代理对象的过程,分为编译期织入、类加载期织入和运行期织入。

三、Spring Boot AOP实战
1、引入依赖
XML 复制代码
<dependency>
   <groupId>org.springframework.boot</groupId>
   <artifactId>spring-boot-starter-aop</artifactId>
   <version>2.7.2</version>
</dependency>

不需要再添加aspectjweaver的依赖了,因为spring-boot-starter-aop包含了aspectjweaver

2、 编写用于拦截的bean

直接定义一个controller

java 复制代码
@RestController
public class AopController {

    @GetMapping("/hello")
    public ResultMap sayHello(){
        ResultMap resultMap = new ResultMap();
        resultMap.setCode(1);
        resultMap.setMessage("响应成功");
        resultMap.setData("Hello World!");

        return resultMap;
    }
}
3、 定义切面

Spring采用@Aspect注解对POJO进行标注,该注解表明该类不仅仅是一个POJO,还是一个切面。切面是切点和通知的结合,那么定义一个切面就需要编写切点和通知。在代码中,只需要添加@Aspect注解即可。

(1)定义切点

切点是通过@Pointcut注解和切点表达式定义的。由于Spring切面粒度最小是达到方法级别,而execution表达式可以用于明确指定方法返回类型,类名,方法名和参数名等与方法相关的部件,并且实际中,大部分需要使用AOP的业务场景也只需要达到方法级别即可,因而execution表达式的使用是最为广泛的。如图是execution表达式的语法:

execution表示在方法执行的时候触发。以"*"开头,表明方法返回值类型为任意类型。然后是全限定的类名和方法名,"*"可以表示任意类和任意方法。对于方法参数列表,可以使用".."表示参数为任意类型。如果需要多个表达式,可以使用"&&"、"||""!"完成与、或、非的操作。

(2)定义通知

通知有五种类型,分别是:

前置通知(@Before):在目标方法调用之前调用通知

后置通知(@After):在目标方法完成之后调用通知

环绕通知(@Around):在被通知的方法调用之前和调用之后执行自定义的方法

返回通知(@AfterReturning):在目标方法成功执行之后调用通知

异常通知(@AfterThrowing):在目标方法抛出异常之后调用通知

java 复制代码
@Aspect
@Component
public class AopAdvice {
    @Pointcut("execution(* com.mq.rabbit.controller.*.*(..))")
    public void pointCut(){

    }

    @Before("pointCut()")
    public void beforeAdvice(){
        System.out.println("\t------前置通知------");
    }

    @After("pointCut()")
    public void afterAdvice(){
        System.out.println("\t------后置通知------");
    }

    @Around("pointCut()")
    public Object aroundAdvice(ProceedingJoinPoint joinPoint)  {
        System.out.println("------环绕通知开始------");
        Object result = null;
        try {
            result = joinPoint.proceed(); //接收Controller的返回值
        }catch (Throwable t){
            t.printStackTrace();
        }
        System.out.println("------环绕通知结束------");
        return result; //将Controller方法的返回值返回
    }
}
4、启动测试

运行项目,在浏览器访问http://localhost:端口号/项目名/hello,控制台输出如图所示:

相关推荐
像少年啦飞驰点、2 分钟前
零基础入门 Spring Boot:从‘Hello World’到可上线微服务的完整学习路径
java·spring boot·web开发·编程入门·后端开发
心 -4 分钟前
全栈实时聊天室(java项目)
java
1104.北光c°23 分钟前
【从零开始学Redis | 第一篇】Redis常用数据结构与基础
java·开发语言·spring boot·redis·笔记·spring·nosql
阿猿收手吧!33 分钟前
【C++】volatile与线程安全:核心区别解析
java·c++·安全
Hui Baby1 小时前
Java SPI 与 Spring SPI
java·python·spring
摇滚侠1 小时前
Maven 教程,Maven 安装及使用,5 小时上手 Maven 又快又稳
java·maven
倔强菜鸟1 小时前
2026.2.2--Jenkins的基本使用
java·运维·jenkins
hai74251 小时前
在 Eclipse 的 JSP 项目中引入 MySQL 驱动
java·mysql·eclipse
瑞雪兆丰年兮1 小时前
[从0开始学Java|第十一天]学生管理系统
java·开发语言
看世界的小gui1 小时前
Jeecgboot通过Maxkey实现单点登录完整方案
java·spring boot·jeecgboot