面试题:Spring AOP底层实现原理

一、AOP与IOC的关系:AOP是IOC的扩展功能

AOP 并非独立运行的技术模块,本质是依托 Spring IOC 容器才能生效的扩展能力。

核心结论:先有 IOC,再有 AOP,AOP 完全依托 Bean 生命周期扩展点 BeanPostProcessor 实现

  • BeanPostProcessor 是 Spring 预留的 Bean 生命周期钩子,允许在 Bean 实例初始化前后,对 Bean 对象进行修改、包装、替换操作。
  • AOP 核心增强逻辑由此实现:目标 Bean 完成创建初始化后,Spring 校验当前类是否匹配切面切点;若需要增强,就通过动态代理生成代理对象,直接替换掉原始目标 Bean,最终把代理对象存入 IOC 容器。

这也是我们通过 @Autowired 注入 Bean 时,拿到的不是原始本尊对象,而是代理对象的根本原因。

二、Spring AOP底层实现完整流程拆解

1. 代理创建:解析切面、切点、增强通知

容器启动阶段,Spring 提前完成 AOP 规则扫描与解析:

  1. 扫描项目所有 @Aspect 切面类;
  2. 解析五大增强通知,同时适配 @Transactional 等基于 AOP 实现的注解;
  3. 解析切点表达式,匹配需要增强的类和方法;
  4. 筛选出所有需要被动态代理增强的目标 Bean。

2. 代理选择:JDK 动态代理 VS CGLIB 动态代理

Spring 根据目标类有无接口,自动选择代理方案:

代理方式 适用场景 核心原理 存在限制
JDK动态代理 目标类实现接口 基于接口生成代理类,依靠 InvocationHandler 拦截方法 只能代理有接口的类
CGLIB动态代理 普通无接口类 通过继承目标类生成子类,重写方法完成拦截增强 无法代理 final 类、final 方法

Spring 内置 CGLIB,无接口类默认使用 CGLIB 代理。

3. 方法拦截入口

调用容器中的 Bean 方法时,不会直接执行本尊业务代码,会被代理拦截:

  • CGLIB 代理:拦截入口为 DynamicAdvisedInterceptor#intercept
  • JDK 动态代理:拦截入口为 InvocationHandler#invoke

两种拦截入口,对应手写动态代理的拦截逻辑,是所有AOP增强的统一入口。

4. 组装拦截器链

Spring 不会单独执行每一个通知,而是将当前方法匹配到的所有切面增强,统一整合为拦截器链

  • 正常流程:前置通知 → 环绕前置 → 目标方法 → 环绕后置 → 返回通知 → 最终后置通知;
  • 异常流程:目标方法报错,触发异常通知,终止正常后置流程;
  • 多切面、多通知场景,依靠拦截器链保证执行顺序可控。

5. 拦截器链递归执行

Spring 底层提供统一顶级接口 MethodInvocation,针对两种代理区分不同实现类,互不混用

  • JDK 动态代理:使用 ReflectiveMethodInvocation
  • CGLIB 动态代理:使用 CglibMethodInvocation

共同运行机制:

  1. 内部维护执行索引,默认从 -1 开始;
  2. 底层不断调用原生 proceed() 方法,递归放行执行下一个拦截器;
  3. 所有前置通知执行完毕后,才会调用本尊目标方法
  4. 目标方法执行完成,逆向执行所有后置通知,最后返回结果。

关键知识点(面试易错纠正)

  1. 业务代码中,只有 @Around 环绕通知 能使用 ProceedingJoinPoint 调用 proceed(),手动控制目标方法执行;

  2. @Before / @After 等普通通知,仅能使用 JoinPoint,没有放行方法;

  3. 普通通知的目标方法调用、拦截器链放行,全部由 Spring 底层自动完成,开发者无法干预。

三、面试高频核心问答

1. 同类内部 this 调用,AOP 为什么失效?

this 指向原始目标本尊对象 ,不是容器中的代理对象,

内部调用会直接绕过代理拦截与拦截器链,AOP、事务全部失效。

解决方式:容器上下文获取代理 Bean、自身注入自我调用。

2. Spring AOP 为什么依赖动态代理?

AOP 核心是无侵入增强 ,不修改业务代码;

动态代理可以在运行时创建代理对象,拦截方法执行,在前后追加通用逻辑,是 AOP 唯一底层实现方案。

3. Spring AOP 五大通知及执行顺序

  • @Before:前置通知
  • @Around:环绕通知(权限最高)
  • @AfterReturning:正常返回通知
  • @AfterThrowing:异常通知
  • @After:最终后置通知

标准顺序:
@Before → 环绕前置 → 目标方法 → 环绕后置 → @AfterReturning/@AfterThrowing → @After

四、整体核心流程总结

  1. 规则解析:容器启动扫描切面、切点、通知,匹配待增强 Bean;
  2. 代理替换 :借助 BeanPostProcessor,创建 JDK/CGLIB 代理,替换本尊 Bean 存入容器;
  3. 方法拦截:调用代理对象方法,进入各自代理的拦截入口;
  4. 链路组装:整合所有切面通知,生成专属拦截器链;
  5. 递归执行 :通过 MethodInvocation 实现类递归调度,执行通知、调用本尊方法,完成AOP增强。
相关推荐
nanxun88619 小时前
记一次诡异的 Docker 容器"串包"故障排查
java
用户15630681035121 小时前
Day01 | Java 基础(Java SE)
java
行者全栈架构师1 天前
Maven dependency:tree 的 8 个高级用法
java·后端
行者全栈架构师1 天前
IDEA 中 Maven 项目的 15 个红色报错快速解决方法
java·后端
令人头秃的代码0_01 天前
mac(m5)平台编译openjdk
java
唐青枫2 天前
Java JDBC 实战指南:从 Connection 到事务和连接池
java
一个做软件开发的牛马2 天前
MyBatis-Plus 从零实战:完整搭建可运行 Demo,BaseMapper 零 SQL、Wrapper 条件构造、分页插件与代码生成器详解
java·后端
用户3721574261352 天前
Java 处理 PDF 图片:提取 PDF 中的图片,并压缩 PDF 图片体积
java
用户3721574261352 天前
Java 打印 Word 文档:从基础打印到高级设置
java
用户3521802454753 天前
当 Prompt 学会"热更新":Spring Boot × Nacos3 AI 实战
java·spring boot·ai编程