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匹配事务方法。
相关推荐
涡能增压发动积16 小时前
同样的代码循环 10次正常 循环 100次就抛异常?自定义 Comparator 的 bug 让我丢尽颜面
后端
云烟成雨TD16 小时前
Spring AI Alibaba 1.x 系列【6】ReactAgent 同步执行 & 流式执行
java·人工智能·spring
Wenweno0o16 小时前
0基础Go语言Eino框架智能体实战-chatModel
开发语言·后端·golang
于慨16 小时前
Lambda 表达式、方法引用(Method Reference)语法
java·前端·servlet
swg32132116 小时前
Spring Boot 3.X Oauth2 认证服务与资源服务
java·spring boot·后端
tyung17 小时前
一个 main.go 搞定协作白板:你画一笔,全世界都看见
后端·go
gelald17 小时前
SpringBoot - 自动配置原理
java·spring boot·后端
殷紫川17 小时前
深入理解 AQS:从架构到实现,解锁 Java 并发编程的核心密钥
java
一轮弯弯的明月17 小时前
贝尔数求集合划分方案总数
java·笔记·蓝桥杯·学习心得
chenjingming66617 小时前
jmeter线程组设置以及串行和并行设置
java·开发语言·jmeter