当前位置: 首页 > news >正文

Spring Boot 切面编程(AOP)详细教程 - 实践

Spring Boot 切面编程(AOP)详细教程 - 实践

Spring Boot 切面编程(AOP)详细教程


一、概述:什么是AOP?为什么需要它?

AOP(Aspect-Oriented Programming)即面向切面编程,是一种与OOP(面向对象编程)互补的编程思想。
简单理解:OOP关注“对象的属性与行为”(比如用户对象有姓名、年龄,能登录),而AOP关注“多个对象/方法的共同行为”(比如所有方法执行前需要记录日志、所有接口调用需要校验权限)。

核心价值:解耦

假设你需要为100个方法添加“日志记录”功能,如果用OOP(在每个方法里写日志代码),会导致代码重复、维护困难。而AOP可以把这些“公共逻辑”抽离成独立的“切面”,自动“切入”到需要的位置,让业务代码保持干净。

二、AOP的核心概念(必须掌握)

术语解释
切面(Aspect)封装公共逻辑的类(比如日志切面、权限切面),用@Aspect注解标记
连接点(JoinPoint)程序执行的某个点(比如方法调用、异常抛出),AOP的“切入点”候选位置
切入点(Pointcut)明确指定要“切入”的连接点(比如“所有UserController类的方法”)
通知(Advice)切面在连接点执行的具体逻辑(比如“方法执行前记录日志”“方法执行后统计耗时”)

三、通知(Advice)的5种类型(最常用!)

Spring AOP通过注解定义通知,控制公共逻辑在“何时执行”。5种通知类型如下:

注解执行时机典型场景
@Before目标方法执行前(不能阻止目标方法执行)日志记录、权限校验
@After目标方法执行后(无论成功/失败都会执行)资源释放、结果汇总
@AfterReturning目标方法成功返回后(失败不执行)结果处理、数据同步
@AfterThrowing目标方法抛出异常后异常日志记录、错误补偿
@Around包裹目标方法(可控制目标方法是否执行性能监控、事务管理

四、使用场景:AOP能解决哪些实际问题?

AOP适合处理跨多个方法/类的公共逻辑,常见场景:

  1. 日志记录:自动记录接口调用参数、返回结果、耗时(比如统计用户登录接口的访问量)。
  2. 权限校验:在敏感方法执行前检查用户是否有权限(比如删除订单前校验是否是订单主人)。
  3. 性能监控:统计方法执行耗时,定位慢接口(比如找出响应时间超过1秒的接口)。
  4. 事务管理:控制数据库事务的开启、提交、回滚(Spring声明式事务的底层就是AOP)。
  5. 参数校验:在方法执行前校验入参是否合法(比如检查用户输入的手机号格式)。

五、代码实现:Spring Boot中如何写AOP?

步骤1:添加依赖(Spring Boot自动集成AOP)

Spring Boot项目默认包含AOP依赖,无需额外添加。如果是手动创建的项目,检查pom.xml是否有:

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

步骤2:定义切面类(核心)

创建一个普通Java类,用@Aspect@Component注解标记(告诉Spring这是一个切面,需要被扫描)。

步骤3:定义切入点(Pointcut)

@Pointcut注解定义“需要切入的方法”,通过切入点表达式指定目标方法。
常用切入点表达式:

步骤4:编写通知(Advice)

在切面类中编写方法,用@Before/@After等注解绑定到切入点,实现公共逻辑。

完整示例:用AOP实现接口调用日志记录
需求:所有UserController的接口被调用时,自动记录“调用时间、方法名、入参、耗时”。
1. 创建自定义注解(可选,用于灵活标记需要记录的方法)

如果只想记录部分方法,可以用注解标记。比如定义@Log注解:

import java.lang.annotation.*;
@Target(ElementType.METHOD) // 作用在方法上
@Retention(RetentionPolicy.RUNTIME) // 运行时生效
public @interface Log {
String desc() default "默认日志"; // 日志描述
}
2. 创建切面类(核心代码)
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;
import java.util.Arrays;
import java.util.Date;
@Aspect // 标记这是一个切面
@Component // 交给Spring管理
public class LogAspect {
/**
* 定义切入点:匹配所有使用@Log注解的方法
* (如果想匹配所有controller方法,改为 execution(* com.example.demo.controller.*.*(..)) )
*/
@Pointcut("@annotation(com.example.demo.annotation.Log)")
public void logPointcut() {}
/**
* @Before:目标方法执行前记录入参
*/
@Before("logPointcut()")
public void beforeLog(JoinPoint joinPoint) {
String methodName = joinPoint.getSignature().getName(); // 获取方法名
Object[] args = joinPoint.getArgs(); // 获取方法入参
System.out.println("【日志-前置】调用时间:" + new Date());
System.out.println("【日志-前置】方法名:" + methodName);
System.out.println("【日志-前置】入参:" + Arrays.toString(args));
}
/**
* @Around:统计方法执行耗时(能控制目标方法是否执行)
*/
@Around("logPointcut()")
public Object aroundLog(ProceedingJoinPoint joinPoint) throws Throwable {
long start = System.currentTimeMillis();
Object result = joinPoint.proceed(); // 执行目标方法(必须调用,否则目标方法不会执行)
long end = System.currentTimeMillis();
System.out.println("【日志-环绕】耗时:" + (end - start) + "ms");
return result; // 返回目标方法的结果
}
/**
* @AfterReturning:目标方法成功返回后记录结果
*/
@AfterReturning(pointcut = "logPointcut()", returning = "result")
public void afterReturningLog(Object result) {
System.out.println("【日志-返回后】返回结果:" + result);
}
}
3. 在业务方法中使用@Log注解(触发切面)
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class UserController {
@Log(desc = "用户登录接口") // 使用自定义注解,触发切面
@PostMapping("/login")
public String login(@RequestBody User user) {
// 模拟登录逻辑
return "登录成功,用户:" + user.getUsername();
}
}
// User类(简单POJO)
class User {
private String username;
private String password;
// get/set方法...
}
4. 测试效果

调用/login接口时,控制台会输出

【日志-前置】调用时间:xxxxxx
【日志-前置】方法名:login
【日志-前置】入参:[User(username=张三, password=123)]
【日志-环绕】耗时:5ms
【日志-返回后】返回结果:登录成功,用户:张三

六、实际业务举例:用AOP实现权限校验

需求:用户调用“删除订单”接口时,必须校验是否是订单的主人。

实现步骤:
  1. 定义@CheckPermission注解(标记需要校验权限的方法)。
  2. 创建权限校验切面,在@Before通知中检查用户是否有权限。
  3. 如果无权限,直接抛出异常,阻止方法执行。
代码示例:
// 1. 自定义注解@CheckPermission
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface CheckPermission {
String value() default "order:delete"; // 权限标识
}
// 2. 权限校验切面
@Aspect
@Component
public class PermissionAspect {
@Pointcut("@annotation(com.example.demo.annotation.CheckPermission)")
public void permissionPointcut() {}
@Before("permissionPointcut()")
public void checkPermission(JoinPoint joinPoint) {
// 从请求中获取当前用户ID(实际项目中可能从Token解析)
Long currentUserId = getCurrentUserIdFromRequest();
// 获取方法参数中的订单ID(假设第一个参数是订单ID)
Object[] args = joinPoint.getArgs();
Long orderId = (Long) args[0];
// 查询订单的主人ID(调用Service)
Long orderOwnerId = orderService.getOrderOwnerId(orderId);
// 校验权限
if (!currentUserId.equals(orderOwnerId)) {
throw new RuntimeException("无权限操作!");
}
}
// 模拟从请求中获取用户ID(实际项目中用Spring Security等框架)
private Long getCurrentUserIdFromRequest() {
return 123L; // 假设当前用户ID是123
}
}
// 3. 在业务方法中使用@CheckPermission
@RestController
public class OrderController {
@CheckPermission("order:delete") // 标记需要校验权限
@PostMapping("/order/delete")
public String deleteOrder(Long orderId) {
orderService.deleteOrder(orderId);
return "订单删除成功";
}
}
效果:

如果当前用户ID(123)与订单主人ID不一致,调用/order/delete接口会直接抛出异常,阻止订单删除操作。

七、总结:AOP的优缺点与注意事项

优点:

  • 代码解耦:公共逻辑与业务逻辑分离,业务代码更干净。
  • 复用性强:一个切面可以应用到多个方法,减少重复代码。
  • 灵活扩展:新增公共逻辑时,只需修改切面,无需改动业务代码。

缺点:

注意事项:

  1. 切入点表达式尽量具体:避免匹配到不需要的方法(比如execution(* *.*(..))会匹配所有方法,导致性能问题)。
  2. @Around通知谨慎使用:必须调用proceed()方法,否则目标方法不会执行。
  3. 异常处理:在@AfterThrowing中捕获异常时,注意不要覆盖业务本身的异常处理逻辑。
  4. 与拦截器的区别:拦截器(Interceptor)是Spring MVC的概念,主要针对HTTP请求;AOP是更底层的面向切面,可作用于任何方法(包括Service、DAO层)。
http://www.jsqmd.com/news/175640/

相关文章:

  • 终极指南:如何深度解析ISO 10303-21 STEP文件格式与工业数据交换
  • Switch引导程序技术解析:hekate自定义固件加载器深度剖析
  • 7个颠覆性教育数据分析技巧:从数据洞察到学习效果提升
  • aarch64启动流程深度剖析:从上电到内核入口的完整指南
  • AD20等长走线调整方法:Altium Designer教程完整示例
  • 2025年评价高的药肥复合肥设备生产线厂家最新推荐权威榜 - 品牌宣传支持者
  • 一键下载600+大模型权重!开源工具助力GPU算力高效利用
  • ModelScope团队贡献:国产大模型生态建设者
  • FactoryBluePrints:戴森球计划终极蓝图库完整使用指南
  • 小白指南:利用screen指令保持SSH远程任务运行
  • 如何在已root的三星设备上绕过Knox限制?完整功能恢复方案
  • Tsuru权限管理系统深度解析:构建企业级RBAC访问控制架构
  • 国产化适配新进展:Ascend NPU全面兼容ms-swift
  • 终极指南:Zen Browser跨平台同步功能全解析
  • FP8与BNB量化详解:极致压缩不影响精度
  • OpenCV实战指南:从零构建计算机视觉应用系统
  • Kronos模型管理终极指南:从本地部署到云端共享
  • 为什么选择bwip-js?5大理由让你爱上这个JavaScript条形码生成库
  • Odyssey.js地图可视化库完全指南:5种惊艳交互效果快速实现
  • DeepSeek-V3.2:企业级AI推理的降本增效新范式
  • Tokens Studio for Figma 完整指南:简单易用的设计令牌管理
  • 界面化操作演示:拖拽完成模型训练全流程
  • 移动端适配技巧:CSS vh 的正确用法
  • 5分钟快速上手Skyvern自动化工具:告别重复性网页操作
  • Wallos个性化主题定制实战指南
  • LivePortrait模型部署终极指南:从12MB到342MB的完整技术选型方案
  • 知识图谱净化工程:从噪声数据到精准检索的蜕变之路
  • Skyvern终极指南:15分钟掌握智能网页自动化技术
  • GraphRag知识图谱数据优化实战:从混乱到清晰的四大核心模块
  • PCSX2 PS2模拟器终极完全指南:从零开始畅玩经典游戏的完整教程