Spring AOP 创建代理对象详情

目录
  • 1 前言
  • 2 创建代理对象
  • 3 AOPContext Aop 上下文
  • 4 AOP 分析汇总
  • 5 JDK 动态代理和 CGLIB 的区别与联系
  • 6 总结

1 前言

在这篇文章中中已经讲述了 AOP 的后置处理器和增强信息的获取,在本文中将继续分享 AOP 创建代理对象和上下文相关的内容。

2 创建代理对象

Spring AOP 使用 JDKProxy 和 CGLIB 两种方式来生成代理对象,具体使用哪一种需要根据 AopProxyFactory 接口的 createProxy 方法中的 AdvisedSupport 中的参数进行确定,默认情况下如果目标类是接口,则使用 jdk 动态代理技术,如果是非接口类,则使用 cglib 来生成代理。

在获取了所有对应 bean 的增强器之后,便可以进行代理的创建了,createProxy 方法。 对于代理类的创建以及处理,spring 委托给 ProxyFactory 进行处理,此方法主要进行初始化操作,并为代理工作做准备。主要分为以下步骤:

  • 1 获取当前代理类的属性。
  • 2 添加代理接口,封装 Advisor 并加入 ProxyFactory 代理工厂中。
  • 3 设置要代理的类,spring 还添加了 customizeProxyFactory ,子类可以在改方法中对 ProxyFactory 进行包装。
  • 4 进行获取代理类的操作。

3 AOPContext Aop 上下文

在最终的代码调用中,还是 JdkDynamicAopProxy 的 invoke 方法和 CglibAopProxy 的 intercept 方法,以 CglibAopProxy 为例,可以看到 AopContext.setCurrentProxy 调用,将当前的对象设置到上下文中,在最后 finally 代码块中会将当前代理对象移除。

AOPContext 的主要方法如下图所示:

4 AOP 分析汇总

综上的逐步分析,最终总结的调用链路图如下所示,分为注册 Bean, 寻找增强器和创建代理等主要环节:

# 注册 bean
invokeBeanFactoryPostProcessors
 @EnableAspectJAutoProxy
 AspectJAutoProxyRegistrar.registerBeanDefinitions
AopConfigUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary
registerOrEscalateApcAsRequired(AnnotationAwareAspectJAutoProxyCreator.class, registry, source);
AopConfigUtils.registerOrEscalateApcAsRequired
# 调用
finishBeanFactoryInitialization
AbstractAutoProxyCreator.postProcessAfterInitialization
AbstractAutoProxyCreator.wrapIfNecessary
AbstractAutoProxyCreator.getAdvicesAndAdvisorsForBean
AbstractAdvisorAutoProxyCreator.getAdvicesAndAdvisorsForBean
AbstractAdvisorAutoProxyCreator.findEligibleAdvisors
AbstractAdvisorAutoProxyCreator.findCandidateAdvisors();
AnnotationAwareAspectJAutoProxyCreator.findCandidateAdvisors
BeanFactoryAspectJAdvisorsBuilder.buildAspectJAdvisors
ReflectiveAspectJAdvisorFactory.getAdvisors
ReflectiveAspectJAdvisorFactory.getPointcut 获取切点的注解信息
new InstantiationModelAwarePointcutAdvisorImpl
instantiateAdvice
ReflectiveAspectJAdvisorFactory.getAdvice 获取 advice
switch
AspectJAroundAdvice
AbstractAspectJAdvice.invokeAdviceMethod
AspectJMethodBeforeAdvice
AspectJAfterAdvice
AspectJAfterReturningAdvice
AspectJAfterThrowingAdvice
AbstractAdvisorAutoProxyCreator.findAdvisorsThatCanApply(candidateAdvisors, beanClass, beanName);
AbstractAutoProxyCreator.createProxy
AbstractAutoProxyCreator.buildAdvisors
proxyFactory.getProxy(classLoader)
DefaultAopProxyFactory.createAopProxy
JdkDynamicAopProxy.getProxy
ObjenesisCglibAopProxy.getProxy

5 JDK 动态代理和 CGLIB 的区别与联系

在 Spring 中,如果说 IOC 和 DI 解决了类与类之间的耦合,那么动态代理则是解决了方法与方法,业务方法和切面逻辑之间的耦合。

JDK 动态代理只能对实现了接口类生成代理,而不能针对类进行代理。Cglib 是针对类进行代理,针对指定的类生成子类,并覆盖方法并进行增强,因为采用的是继承方式,所代理的类和方法不能是 final 修饰(final 的类或方法,是无法继承的)。

jdk 动态代理:创建一个实现 InvocationHandler 接口的拦截器,使用 Proxy.newProxyInstance() 反射机制生成一个代理对象,在具体调用方法之前先调用 InvokeHandler.invoke 来处理。

Cglib 动态代理:利用 ASM 字节码增强框架,对代理对象生成的 class 文件加载进来,通过修改字节码生成子类进行代理。

jdk 和 cglib 使用方法的区别:

  • 1 目标对象如果实现了接口,则默认使用 jdk 动态代理。
  • 2 如果目标对象实现了接口,可以强制使用 cglib。
  • 3 如果目标对象没有实现接口,则必须使用 cglib, spring 会自动在 jdk 和 cglib 代理之间进行切换。

综上, jdk 代理只能针对实现接口的类生成代理,而不能针对类。cglib 只能针对类进行代理。

6 总结

在本文中主要讲述了后置处理器和 AOP 的实现原理,以及 jdk 动态代理和 cglib 代理之间的区别,主要涉及增强获取和代理类的获取,以及后置处理器的理解。

到此这篇关于Spring AOP 创建代理对象详情的文章就介绍到这了,更多相关Spring AOP 内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Spring基于ProxyFactoryBean创建AOP代理

    Spring 通知类型 通过前面的学习可以知道,通知(Advice)其实就是对目标切入点进行增强的内容,Spring AOP 为通知(Advice)提供了 org.aopalliance.aop.Advice 接口. Spring 通知按照在目标类方法的连接点位置,可以分为以下五种类型,如表 1 所示. 表 1 Spring 通知的 5 种类型 名称 说明 org.springframework.aop.MethodBeforeAdvice(前置通知) 在方法之前自动执行的通知称为前置通知,可以

  • 带你了解如何使用Spring基于ProxyFactoryBean创建AOP代理

    目录 1 基础 2 JavaBean属性 3 JDK和CGLIB代理 总结 若使用 Spring IoC 容器(ApplicationContext或BeanFactory)作为你的业务对象(你也应该这么做!),你会想使用 Spring AOP FactoryBean的一种. 工厂 bean 引入了中间层,让它创建不同类型的对象. 在Spring创建 AOP 代理的基本方式是使用 org.springframework.aop.framework.ProxyFactoryBean.这可以完全控制

  • Spring-AOP自动创建代理之BeanNameAutoProxyCreator实例

    实例 代码已托管到Github-> https://github.com/yangshangwei/SpringMaster 在 Spring-AOP 静态普通方法名匹配切面 案例中,我们通过配置两个ProxyFactoryBean分别为waiter和seller的Bean创建代理对象, 如下 <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.spring

  • 这一次搞懂Spring代理创建及AOP链式调用过程操作

    前言 AOP,也就是面向切面编程,它可以将公共的代码抽离出来,动态的织入到目标类.目标方法中,大大提高我们编程的效率,也使程序变得更加优雅.如事务.操作日志等都可以使用AOP实现.这种织入可以是在运行期动态生成代理对象实现,也可以在编译期.类加载时期静态织入到代码中.而Spring正是通过第一种方法实现,且在代理类的生成上也有两种方式:JDK Proxy和CGLIB,默认当类实现了接口时使用前者,否则使用后者:另外Spring AOP只能实现对方法的增强. 正文 基本概念 AOP的术语很多,虽然

  • Spring AOP 创建代理对象详情

    目录 1 前言 2 创建代理对象 3 AOPContext Aop 上下文 4 AOP 分析汇总 5 JDK 动态代理和 CGLIB 的区别与联系 6 总结 1 前言 在这篇文章中中已经讲述了 AOP 的后置处理器和增强信息的获取,在本文中将继续分享 AOP 创建代理对象和上下文相关的内容. 2 创建代理对象 Spring AOP 使用 JDKProxy 和 CGLIB 两种方式来生成代理对象,具体使用哪一种需要根据 AopProxyFactory 接口的 createProxy 方法中的 Ad

  • Spring AOP 与代理的概念与使用

    一.AOP 的基本概念 1.1 什么是 AOP Aspect Oriented Programming,面向切面编程. 就跟我们说 OOP 是面向对象一样,AOP 是面向切面的.切面是分散在应用中的一个标准代码或功能.切面通常与实际的业务逻辑不同(例如,事务管理).每个切面专注于一个特定的环切功能. 这里的切面呢,可以理解为横切.比如在所有的 DAO 层方法上加上一个同样的切面,功能是记录日志:又或者在某个接口上应用一个切面,作用是检查权限. AOP 是基于代理来实现的.而代理又分为静态代理和动

  • Spring aop 如何通过获取代理对象实现事务切换

    Spring aop 获取代理对象实现事务切换 在项目中,涉及到同一个类中一个方法调用另外一个方法,并且两个方法的事务不相关, 这里面涉及到一个事务切换的问题,一般的方法没问题,根据通过aop注解在方法上通过加注解标识, 答案是: 通过spring aop类里面的AopContext类获取当前类的代理对象, 这样就能切换对应的事务管理器了,具体做法如下: (1).在applicationContext.xml文件中配置如下: <!-- 开启暴露Aop代理到ThreadLocal支持 --> &

  • 创建动态代理对象bean,并动态注入到spring容器中的操作

    使用过Mybatis的同学,应该都知道,我们只需要编写mybatis对应的接口和mapper XML文件即可,并不需要手动编写mapper接口的实现.这里mybatis就用到了JDK动态代理,并且将生成的接口代理对象动态注入到Spring容器中. 这里涉及到几个问题.也许有同学会有疑问,我们直接编写好类,加入@Component等注解不是可以注入了吗?或者在配置类(@Configuration)中直接声明该Bean类型不也可以注入吗? 但具体到mybatis,这里我们用的是接口.由于spring

  • Spring AOP中的JDK和CGLib动态代理哪个效率更高?

    一.背景 今天有小伙伴面试的时候被问到:Spring AOP中JDK 和 CGLib动态代理哪个效率更高? 二.基本概念 首先,我们知道Spring AOP的底层实现有两种方式:一种是JDK动态代理,另一种是CGLib的方式. 自Java 1.3以后,Java提供了动态代理技术,允许开发者在运行期创建接口的代理实例,后来这项技术被用到了Spring的很多地方. JDK动态代理主要涉及java.lang.reflect包下边的两个类:Proxy和InvocationHandler.其中,Invoc

  • 详解在Spring中如何自动创建代理

    Spring 提供了自动代理机制,可以让容器自动生成代理,从而把开发人员从繁琐的配置中解脱出来 . 具体是使用 BeanPostProcessor 来实现这项功能. 1 BeanPostProcessor BeanPostProcessor 代理创建器的实现类可以分为 3 类: 类型 实现类 基于 Bean 配置名规则 BeanNameAutoProxyCreator 基于 Advisor 匹配规则 DefaultAdvisorAutoProxyCreator 基于 Bean 中的 Aspect

  • Spring AOP里的静态代理和动态代理用法详解

    什么是代理? 为某一个对象创建一个代理对象,程序不直接用原本的对象,而是由创建的代理对象来控制原对象,通过代理类这中间一层,能有效控制对委托类对象的直接访问,也可以很好地隐藏和保护委托类对象,同时也为实施不同控制策略预留了空间 什么是静态代理? 由程序创建或特定工具自动生成源代码,在程序运行前,代理类的.class文件就已经存在 通过将目标类与代理类实现同一个接口,让代理类持有真实类对象,然后在代理类方法中调用真实类方法,在调用真实类方法的前后添加我们所需要的功能扩展代码来达到增强的目的. 优点

  • spring aop代理控制的操作方式

    目录 默认代理行为 修改代理行为 spring aop可通过参数proxyTargetProxy控制创建代理的方式 proxyTargetProxy=true:强制使用cglib代理proxyTargetProxy=false:目标实现类实现了接口使用jdk,没有实现接口则使用cglib springboot 默认代理行为 # 通过参数spring.aop.proxy-target-proxy控制 1.x:proxy-target-proxy=false 2.x:proxy-target-pro

  • Spring AOP底层原理及代理模式

    目录 Spring AOP底层原理代理模式 一.什么是 AOP 二.AOP 底层原理 1. 什么是代理? 2. 什么是静态代理 3. 什么是动态代理 Spring AOP底层原理代理模式 一.什么是 AOP AOP 就是面向切面编程,是 OOP(面向对象编程)的延续. 利用 AOP 可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序可用性,同时也提高了开发效率. 通俗一点说,不用修改原代码,可以给原代码增加新的功能. 二.AOP 底层原理 AOP 底层原理是使用动

  • Spring AOP 对象内部方法间的嵌套调用方式

    目录 Spring AOP 对象内部方法间的嵌套调用 我们先定义一个接口 以及此接口的一个实现类 增加AOP处理 同一对象内的嵌套方法调用AOP失效原因分析 举一个同一对象内的嵌套方法调用拦截失效的例子 原因分析 解决方案 Spring AOP 对象内部方法间的嵌套调用 前两天面试的时候,面试官问了一个问题,大概意思就是一个类有两个成员方法 A 和 B,两者都加了事务处理注解,定义了事务传播级别为 REQUIRE_NEW,问 A 方法内部直接调用 B 方法时能否触发事务处理机制. 答案有点复杂,

随机推荐