Spring Aop底层源码实现(一)

动态代理基础与实现

  • 动态代理的作用
    • 在不修改目标类的前提下,为目标类的方法添加额外逻辑。
  • 两种代理方式:JDK动态代理与CGLIB代理
    • JDK动态代理 :基于接口,使用java.lang.reflect.Proxy
    • CGLIB代理:基于继承,适用于没有接口的目标类。
  • Spring对代理机制的封装
    • Spring内部使用ProxyFactory统一处理代理逻辑,屏蔽JDK与CGLIB的差异。

CGLIB代理演示与代理逻辑实现

  • CGLIB代理基本用法
    • 使用Enhancer类创建代理。
    • 设置目标类、回调逻辑(MethodInterceptor)。
  • MethodInterceptor接口作用
    • intercept方法中定义代理逻辑。
    • 手动调用MethodProxy.invokeSuper执行目标方法。
  • 代理对象与目标对象区别
    • Object参数为代理对象,MethodProxy用于调用目标方法。
  • 常见错误:死循环问题
    • 若在intercept中调用method.invoke(obj, args)会引发死循环。
    • 正确方式应使用methodProxy.invokeSuper

JDK动态代理示例

  • JDK代理基本用法
    • 使用Proxy.newProxyInstance创建代理。
    • 实现InvocationHandler接口定义代理逻辑。
  • JDK代理限制
    • 必须基于接口,被代理类需实现接口。
  • 代理对象类型
    • 代理对象类型为接口类型,不能直接转换为目标类。

ProxyFactory统一代理创建机制

  • ProxyFactory的作用
    • 封装JDK与CGLIB代理,对外提供统一API。
    • 可通过配置指定使用哪种代理方式。
  • 添加代理逻辑:Advice
    • 使用addAdvice方法添加如MethodBeforeAdviceAfterReturningAdviceThrowsAdvice等。
    • 每种Advice对应不同执行时机(方法前、方法后、异常后)。
  • 执行顺序与责任链模式
    • 多个Advice按添加顺序构成责任链。
    • 每个Advice执行后自动调用下一个节点,最终执行目标方法。

Advisor与Pointcut:更细粒度的代理控制

  • Advisor = Pointcut + Advice
    • Pointcut:定义代理逻辑应用的方法范围(如匹配方法名、参数、类等)。
    • Advice:定义具体代理逻辑。
  • Pointcut匹配逻辑
    • 可自定义匹配规则,如通过方法名、异常类型等。
    • Spring提供NameMatchMethodPointcutAspectJExpressionPointcut等实现。
  • Advisor的使用场景
    • 用于指定某个Advice只在特定方法上生效,而非所有方法。

自动代理创建器:BeanPostProcessor实现

  • BeanPostProcessor机制
    • 在Bean初始化阶段介入,创建代理对象并替换原始Bean。
  • BeanNameAutoProxyCreator
    • 根据Bean名称匹配,自动为匹配的Bean创建代理。
  • DefaultAdvisorAutoProxyCreator
    • 自动扫描Spring容器中所有Advisor,并为匹配Pointcut的Bean创建代理。
    • 基于BeanPostProcessor机制实现,无需手动配置代理逻辑。

Spring AOP与AspectJ的关系

  • AOP是一种编程思想
    • 通过切面(Aspect)将横切逻辑与业务逻辑分离。
  • Spring AOP与AspectJ区别
    • Spring AOP:基于动态代理,运行期织入,仅支持方法级别的切面。
    • AspectJ:编译期织入,支持字段、构造器、注解等更细粒度切面。
  • Spring AOP对AspectJ的兼容
    • Spring AOP使用了AspectJ的注解(如@Before@After),但底层实现基于动态代理。
    • 使用Spring AOP需引入AspectJ的注解依赖(如aspectjweaver)。
    • Spring通过解析AspectJ风格的切面,生成对应的Advisor和Pointcut。

Spring AOP核心组件与注解支持

  • 核心组件
    • Aspect:切面,包含多个Pointcut和Advice。
    • JoinPoint:连接点,即被拦截的方法。
    • Pointcut:定义哪些方法需要增强。
    • Advice:定义增强逻辑及执行时机。
  • 注解驱动的AOP配置
    • 使用@Aspect定义切面类。
    • 使用@Before@After@Around等定义增强逻辑。
    • 使用@EnableAspectJAutoProxy启用AOP自动代理。

Spring AOP源码实现原理

  • EnableAspectJAutoProxy注解作用
    • 导入AspectJAutoProxyRegistrar,注册AnnotationAwareAspectJAutoProxyCreator
  • AnnotationAwareAspectJAutoProxyCreator
    • 继承AbstractAutoProxyCreator,负责扫描Advisor并创建代理。
    • 在Bean生命周期中通过postProcessAfterInitialization创建代理对象。
  • 切面解析机制
    • 解析@Aspect注解类,提取其中的@Before@After等注解。
    • 将每个注解解析为对应的AdvicePointcut,封装为Advisor
  • 代理创建过程
    • 通过ProxyFactory创建代理对象。
    • 使用JdkDynamicAopProxyCglibAopProxy生成代理。

Spring AOP与Lazy加载、@Transactional等机制的结合

  • Lazy加载原理
    • 使用ProxyFactory生成代理对象,延迟初始化真实Bean。
  • @Transactional事务管理
    • 通过AOP在方法前后添加事务控制逻辑。
    • 使用TransactionInterceptor作为Advice,结合TransactionAttributeSource匹配事务方法。
相关推荐
耗子会飞1 小时前
小白学习springboot项目如何连接RocketMQ
后端·rocketmq
ZTrainWilliams1 小时前
swagger-mcp-toolkit 让 AI编辑器 更快“读懂并调用”你的接口
前端·后端·mcp
cylgdzz1112 小时前
PageIndex:一种不靠向量检索的长文档 RAG 实现思路
后端
weixin_456321642 小时前
Java架构设计:Redis持久化方案整合实战
java·开发语言·redis
Later2 小时前
Apache Doris 深度讲解:从核心概念到实战项目
后端
攒了一袋星辰2 小时前
SequenceGenerator高并发有序顺序号生成中间件 - 架构设计文档
java·后端·spring·中间件·架构·kafka·maven
码农刚子2 小时前
字符串拼接用“+”还是 StringBuilder?别再凭感觉写了
后端·代码规范
lzp07912 小时前
SpringBoot3.3.0集成Knife4j4.5.0实战
java
茶杯梦轩2 小时前
面试常问:DNS,CDN,Cookie,Session和Token详解及实战避坑指南
后端·网络协议·面试