Spring AOP

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: Spring AOP

AOP (Aspect Oriented Programming) 面向切面编程

AOP 是一种思想, 是对某一类事情的集中处理

AOP 的作用

在程序运行期间, 在不改变源代码的基础上, 对已有方法进行增强 (无侵入性: 解耦)

AOP 的常见用途

  • 统一日志思想
  • 统一执行方法计时
  • 事务的开启和提交
  • 统一数据返回格式
  • 统一异常处理

Spring AOP

Spring AOP 是 AOP 思想的一种实现

AOP 思想有很多实现 : Spring Boot 统一功能处理 , Spring AOP , AspectJ , CGLIB …

Spring AOP 的实现方式


  1. 基于注解 @Aspect 实现 (常用)
  2. 基于 Spring API (xml 文件) 方式实现
  3. 基于代理实现

Spring Boot 统一功能处理 和 Spring AOP

Spring Boot 统一功能处理 和 Spring AOP 是 相辅相成 的技术

像拦截器作用维度是 url(一次请求和响应), @ControllerAdvice 的应用场景主要是全局异常处理, 数据绑定, 数据预处理.

Spring AOP 的作用维度则可以更加细致 (包, 类, 方法, 参数 …), 能够实现更复杂的业务逻辑

Spring AOP:

Spring AOP 是 Spring 框架的一个模块,用于支持面向切面编程。它通过在方法执行前、执行后或抛出异常时动态地添加横切逻辑,例如日志记录、性能监控、事务管理等。AOP 的主要目的是在不修改原始代码的情况下,通过将横切逻辑与业务逻辑分离,提高代码的模块性和可维护性。

Spring Boot:

Spring Boot 是 Spring 框架的一个扩展,旨在简化基于 Spring 的应用程序的开发和部署。它提供了自动配置、快速构建、嵌入式服务器等特性,使得开发者可以更加便捷地创建独立的、生产级别的 Spring 应用程序。

功能统一处理:

在实际应用中,Spring AOP 可以用于统一处理某些横切关注点,如日志记录、权限控制等,而 Spring Boot 可以用于统一处理应用程序的配置、异常处理、安全性等方面的功能。因此,虽然 Spring AOP 和 Spring Boot 在实现功能统一处理上有一定的重叠,但它们更多地是在不同层面上为应用程序提供支持,而不是直接相关的概念或功能模块。

综上所述,Spring AOP 和 Spring Boot 都可以用于实现功能的统一处理,但它们是不同的模块,各自在不同的层面提供支持,没有直接的关联关系。


Spring AOP 的原理

Spring AOP 基于 动态代理 实现

代理模式(委托模式)

想了解动态代理, 就得先了解代理模式

代理模式 : 为其他对象提供一种代理, 以控制对这个对象的访问. 它的作用是通过一个代理类, 让我们在调用目标方法的时候, 不再是直接调用目标方法 (有可能参数不适配, 返回值不准确 …) , 而是通过调用代理类, 代理类里面再调用目标方法的方式, 间接调用

代理模式主要角色

  1. Subject : 业务接口类. (不一定有)
  2. RealSubject : 业务实现类. 具体的业务执行, 即被代理对象
  3. Proxy : 代理类

代理模式分类


代理模式分为静态代理和动态代理


静态代理 : 在程序运行期代理类的 .class 文件已经存在, 是被写死不能修改的代理

动态代理 : 在程序运行时, 运用 反射机制 动态创建而成的代理

Java 中动态代理有两种实现: JDK 动态代理 & CGBIG 动态代理

JDK 动态代理, 只能代理接口, 不能代理普通类

CGLIB 动态代理, 能够代理接口和普通类

什么时候使用什么代理?

Spring AOP 中, 两种动态代理模式都使用了, 具体使用哪种方式, 还是依据 版本和配置 实现:

  • 在配置文件中, 通过属性 proxyTargetClass 设置使用哪种代理方式
  • Spring Boot 2.X 版本 之前, 该属性默认为 false, 即目标对象实现接口, 则使用 jdk 静态代理, 目标方法未实现接口 (只有实现类), 则使用 cglib 动态代理
  • Spring Boot 2.X 版本 之后, 该属性默认为 true, 无论目标对象是否实现接口, 都使用 cglib 动态代理

基于 @Aspect 注解实现 Spring AOP

首先要引入依赖

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

使用 Spring AOP 编写, 对调用的方法计时操作

@Slf4j
@Component
@Aspect
public class SpringAOP {
    @Pointcut("execution(* com.zrj.mybatisreview.*.*(..))")
    public void pt(){}

    //@Around("execution(* com.zrj.mybatisreview.*.*(..))")
    @Around("pt()")
    public Object recordTime(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        // 使用该方法记录 每个方法被调用的时长
        // 记录方法执行开始时间
        long begin = System.currentTimeMillis();
        // 执行原始方法
        Object result = proceedingJoinPoint.proceed();
        // 记录方法执行结束时间
        long end = System.currentTimeMillis();
        // 日志打印方法执行时长
        log.info(proceedingJoinPoint.getSignature() + "执行耗时: {}ms", end - begin);
        // 返回原始方法运行结果 (不返回运行结果, 就相当于代码没执行 ...)
        return result;
    }
    
    @Before("pt()")
    public void beforeAOP() {
        log.info("hello world!");
    }
}


Spring AOP 中的一些概念

切点(Pointcut)

也叫做 “切入点”

切点就是 一组规则 (好抽象), 通过切点表达式, 告诉程序应该对哪些方法进行功能增强

@Pointcut 提取出公共的切点表达式, 可供重复使用

连接点(Join Point)

满足切面表达式规则的所有方法, 都成为连接点

可以理解为: 切点是保存了众多连接点的一个集合

通知(Advice)

通知就是具体要做的内容, 指哪些重复的逻辑, 也就是共性功能

上述代码为例就是对每个方法进行计时操作, 并打印日志

切面(Aspect)

切面 = 切点 + 通知

通过切点能够描述出 AOP 的作用范围, 通知则告诉 对切点具体要执行的操作

切面即包含了通知逻辑的定义, 也包含了连接点的定义

切面所在的类, 一般称为 切面类 (被 @Aspect 注解标注的类)

一个切面类可以包含多个切点


切面优先级 @Order

  • 使用场景: 多切面, 且具有 相同类型 通知情况下, 指定不同通知的执行顺序
  • 数字越小, 优先级越高, 同种类型通知, 先执行优先级高的, 在执行优先级低的


Spring AOP 的通知类型

@Around : 环绕通知

@Before : 前置通知

@After : 后置通知

@AfterReturning : 返回后通知

@AfterThrowing : 异常后通知

通知执行顺序

通知注意事项

  • @Around 环绕通知需要主动调用 proceedingJoinPoint.proceed() 方法来让原始方法执行, 其他通知则不需要考虑目标方法的执行
  • @Around 环绕通知方法的返回值, 必须指定为 Object 类型, 来接受原始方法的返回值, 否则原始方法执行完毕, 是获取不到返回值的

切点表达式

  1. execution() : 根据方法签名来 定义切点规则
  1. @annotation() : 根据注解来匹配 定义切点规则

execution()

execution() 是最常见的切点表达式, 用来匹配方法, 语法为:

execution(<访问修饰符> <返回类型> <包名.类名.方法(方法参数)> <异常>)

切点表达式支持通配符匹配

  • * 匹配任意一个元素
  • .. 匹配任意个元素

@annotation

@annotation 常用来匹配多个无规则方法

(一个类中一部分方法需要匹配, 一部分方法不需要匹配, 此时再使用 execution() 就不太方便了)

@annotation 使用流程

  1. 编写自定义注解 (如果使用本来就有的注解 [eg: @Controller], 可以省略这一步)
  2. 使用 @annotation 表达式来描述切点
  3. 再连接点的方法上添加自定义注解
// 注解类代码

package com.zrj.mybatisreview;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;


@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface TimeRecord {
    // 使用本注解对需要进行计时的方法进行标注
}
// 切面类代码

@Slf4j
@Aspect
@Component
public class TimeRecord {
  // @annotation() 中的参数是注解的位置(包名.注解名)
    @Before("@annotation(com.zrj.mybatisreview.TimeRecord)") 
    public void before() {
        log.info("TimeRecord -> begin");
    }

    @After("@annotation(com.zrj.mybatisreview.TimeRecord)")
    public void after() {
        log.info("TimeRecord -> after");
    }
}

对想要计时的方法添加 自定义注解, 该方法就会在调用的时候自动执行 AOP 逻辑

相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
目录
相关文章
|
2月前
|
Java
Spring5入门到实战------9、AOP基本概念、底层原理、JDK动态代理实现
这篇文章是Spring5框架的实战教程,深入讲解了AOP的基本概念、如何利用动态代理实现AOP,特别是通过JDK动态代理机制在不修改源代码的情况下为业务逻辑添加新功能,降低代码耦合度,并通过具体代码示例演示了JDK动态代理的实现过程。
Spring5入门到实战------9、AOP基本概念、底层原理、JDK动态代理实现
|
6天前
|
设计模式 Java 测试技术
spring复习04,静态代理动态代理,AOP
这篇文章讲解了Java代理模式的相关知识,包括静态代理和动态代理(JDK动态代理和CGLIB),以及AOP(面向切面编程)的概念和在Spring框架中的应用。文章还提供了详细的示例代码,演示了如何使用Spring AOP进行方法增强和代理对象的创建。
spring复习04,静态代理动态代理,AOP
|
3月前
|
Java Spring
在Spring Boot中使用AOP实现日志切面
在Spring Boot中使用AOP实现日志切面
|
20天前
|
Java 数据库连接 数据库
Spring基础3——AOP,事务管理
AOP简介、入门案例、工作流程、切入点表达式、环绕通知、通知获取参数或返回值或异常、事务管理
Spring基础3——AOP,事务管理
|
2月前
|
XML Java 数据格式
Spring5入门到实战------11、使用XML方式实现AOP切面编程。具体代码+讲解
这篇文章是Spring5框架的AOP切面编程教程,通过XML配置方式,详细讲解了如何创建被增强类和增强类,如何在Spring配置文件中定义切入点和切面,以及如何将增强逻辑应用到具体方法上。文章通过具体的代码示例和测试结果,展示了使用XML配置实现AOP的过程,并强调了虽然注解开发更为便捷,但掌握XML配置也是非常重要的。
Spring5入门到实战------11、使用XML方式实现AOP切面编程。具体代码+讲解
|
2月前
|
缓存 Java 开发者
Spring高手之路22——AOP切面类的封装与解析
本篇文章深入解析了Spring AOP的工作机制,包括Advisor和TargetSource的构建与作用。通过详尽的源码分析和实际案例,帮助开发者全面理解AOP的核心技术,提升在实际项目中的应用能力。
23 0
Spring高手之路22——AOP切面类的封装与解析
|
2月前
|
安全 Java 开发者
Java 新手入门:Spring 两大利器IoC 和 AOP,小白也能轻松理解!
Java 新手入门:Spring 两大利器IoC 和 AOP,小白也能轻松理解!
31 1
|
2月前
|
Java Spring
Spring的AOP组件详解
该文章主要介绍了Spring AOP(面向切面编程)组件的实现原理,包括Spring AOP的基础概念、动态代理模式、AOP组件的实现以及Spring选择JDK动态代理或CGLIB动态代理的依据。
Spring的AOP组件详解
|
2月前
|
Java API Spring
Spring Boot 中的 AOP 处理
对 Spring Boot 中的切面 AOP 做了详细的讲解,主要介绍了 Spring Boot 中 AOP 的引入,常用注解的使用,参数的使用,以及常用 api 的介绍。AOP 在实际项目中很有用,对切面方法执行前后都可以根据具体的业务,做相应的预处理或者增强处理,同时也可以用作异常捕获处理,可以根据具体业务场景,合理去使用 AOP。
|
2月前
|
Java Spring XML
掌握面向切面编程的秘密武器:Spring AOP 让你的代码优雅转身,横切关注点再也不是难题!
【8月更文挑战第31天】面向切面编程(AOP)通过切面封装横切关注点,如日志记录、事务管理等,使业务逻辑更清晰。Spring AOP提供强大工具,无需在业务代码中硬编码这些功能。本文将深入探讨Spring AOP的概念、工作原理及实际应用,展示如何通过基于注解的配置创建切面,优化代码结构并提高可维护性。通过示例说明如何定义切面类、通知方法及其应用时机,实现方法调用前后的日志记录,展示AOP在分离关注点和添加新功能方面的优势。
38 0
下一篇
无影云桌面