Spring 源码阅读 56:创建 ProxyFactory

简介: 本文分析了用于创建 AOP 代理的后处理器在得到当前 Bean 实例对应的所有增强逻辑之后,创建代理的第一步,也就是先创建一个 ProxyFactory 工厂对象。

基于 Spring Framework v5.2.6.RELEASE

概述

前几篇文章,介绍了 Spring 在创建 AOP 代理之前,查找与 Bean 实例匹配的 Advisor 的过程,本文开始分析后续的过程,主要是代理对象创建的过程。

找到 Advisor 后的流程

我们再回到 AbstractAutoProxyCreator 的wrapIfNecessary方法。

// org.springframework.aop.framework.autoproxy.AbstractAutoProxyCreator#wrapIfNecessaryprotectedObjectwrapIfNecessary(Objectbean, StringbeanName, ObjectcacheKey) {
// 省略部分逻辑// Create proxy if we have advice.Object[] specificInterceptors=getAdvicesAndAdvisorsForBean(bean.getClass(), beanName, null);
if (specificInterceptors!=DO_NOT_PROXY) {
this.advisedBeans.put(cacheKey, Boolean.TRUE);
Objectproxy=createProxy(
bean.getClass(), beanName, specificInterceptors, newSingletonTargetSource(bean));
this.proxyTypes.put(cacheKey, proxy.getClass());
returnproxy;
   }
this.advisedBeans.put(cacheKey, Boolean.FALSE);
returnbean;
}

通过getAdvicesAndAdvisorsForBean获取到的与当前 Bean 实例匹配的 Advisor 列表,会被赋值给变量specificInterceptors,它是一个 Object 列表。也就是说,可以用来增强目标方法的逻辑,也被称作拦截器,而且不一定都是 Advisor 类型。

接下来会判断specificInterceptors是否为空,如果为空的话,说明当前的 Bean 实例不需要被代理,直接把 Bean 实例返回就可以了。如果specificInterceptors不为空的话,则为其创建代理对象,代理对象的创建通过createProxy方法来完成。

我们进入createProxy方法。

image.png

从整体来看,这个方法的主要逻辑就是,创建一个 ProxyFactory 工厂对象,然后对其进行配置,最后调用它的getProxy方法获取代理对象。可见 ProxyFactory 类非常重要,我们先来分析一下这个类。

ProxyFactory 类

先看它的类继承关系。

image.png

在 ProxyFactory 源码中,只声明了几个构造方法和几个getProxy重载方法,它的其他逻辑都是继承了父类中的实现。因此,需要对它的继承关系大概有个印象,后面的分析中遇到了它的父类或者实现的接口,就知道它们与 ProxyFactory 之间的关系了。

ProxyFactory 的创建

我们回到createProxy方法中,一步一步分析具体的逻辑。

保留目标 Bean 的类型信息

if (this.beanFactoryinstanceofConfigurableListableBeanFactory) {
AutoProxyUtils.exposeTargetClass((ConfigurableListableBeanFactory) this.beanFactory, beanName, beanClass);
   }

如果当前的 Bean 容器是 ConfigurableListableBeanFactory 类型,则执行 AutoProxyUtils 的exposeTargetClass方法。我们进入到这个方法当中。

// org.springframework.aop.framework.autoproxy.AutoProxyUtils#exposeTargetClassstaticvoidexposeTargetClass(
ConfigurableListableBeanFactorybeanFactory, @NullableStringbeanName, Class<?>targetClass) {
if (beanName!=null&&beanFactory.containsBeanDefinition(beanName)) {
beanFactory.getMergedBeanDefinition(beanName).setAttribute(ORIGINAL_TARGET_CLASS_ATTRIBUTE, targetClass);
   }
}

这里是给容器中,当前 Bean 实例对应的 BeanDefinition 设置一个属性,保留被代理的 Bean 实例的原始类型信息。

回到createProxy方法继续往下看代码。

通过构造方法创建 ProxyFactory

ProxyFactoryproxyFactory=newProxyFactory();

可以看到 ProxyFactory 是通过构造方法来创建的,进入 ProxyFactory 的无参构造方法查看。

// org.springframework.aop.framework.ProxyFactory#ProxyFactory()publicProxyFactory() {
}

没有任何内容,不过根据 Java 构造方法的调用机制,我们还需要去它的父类中查找逻辑。

在父类 ProxyCreatorSupport 的无参构造方法中,有如下内容。

// org.springframework.aop.framework.ProxyCreatorSupport#ProxyCreatorSupport()publicProxyCreatorSupport() {
this.aopProxyFactory=newDefaultAopProxyFactory();
}

这里初始化了aopProxyFactory成员变量,它的类型是 DefaultAopProxyFactory,先留个印象,这个成员变量后面会用到。

再顺着继承关系往上找,可以看到 AdvisedSupport 的无参构造方法中,有如下逻辑。

// org.springframework.aop.framework.AdvisedSupport#AdvisedSupport()publicAdvisedSupport() {
this.methodCache=newConcurrentHashMap<>(32);
}

这里被初始化的成员变量methodCache应该是一个存放跟方法有关的缓存信息的集合,我们之后肯定会遇到,到时候再看它的作用。

在 ProxyFactory 被创建的构造方法中,就执行了这些逻辑,也就是初始化了两个声明在父类中的成员变量。在执行完构造方法之后,又调用了一个方法。

proxyFactory.copyFrom(this);

传入的参数是当前的后处理器对象,我们进入方法查看一下。

// org.springframework.aop.framework.ProxyConfig#copyFrompublicvoidcopyFrom(ProxyConfigother) {
Assert.notNull(other, "Other ProxyConfig object must not be null");
this.proxyTargetClass=other.proxyTargetClass;
this.optimize=other.optimize;
this.exposeProxy=other.exposeProxy;
this.frozen=other.frozen;
this.opaque=other.opaque;
}

这个方法声明在 ProxyFactory 的父类 ProxyConfig,从父类的类名看应该是封装一些代理的配置信息,在copyFrom方法中,逻辑也比较简单,就是把后处理器中的一些配置,赋值给刚刚创建的 ProxyFactory 对象,这些应该都是后面创建代理对象时会用到的配置信息,我们在后面分析中遇到了再具体分析。

至此,就完成了 ProxyFactory 代理工厂对象的基本创建,后续就是对 ProxyFactory 的各项参数进行配置,也就是在创建代理对象前对proxyFactory工厂对象进行初始化。配置的内容比较多,我会放在下一篇进行分析。

总结

本文分析了用于创建 AOP 代理的后处理器在得到当前 Bean 实例对应的所有增强逻辑之后,创建代理的第一步,也就是先创建一个 ProxyFactory 工厂对象。下一篇,我会继续分析后续对工厂对象的各项属性配置,也就是初始化的过程。

目录
相关文章
|
20天前
|
Java 应用服务中间件 Nacos
Spring Cloud 常用各个组件详解及实现原理(附加源码+实现逻辑图)
Spring Cloud 常用各个组件详解及实现原理(附加源码+实现逻辑图)
31 0
|
23天前
|
监控 数据可视化 安全
一套成熟的Spring Cloud智慧工地平台源码,自主版权,开箱即用
这是一套基于Spring Cloud的智慧工地管理平台源码,具备自主版权,易于使用。平台运用现代技术如物联网、大数据等改进工地管理,服务包括建设各方,提供人员、车辆、视频监控等七大维度的管理。特色在于可视化管理、智能报警、移动办公和分布计算存储。功能涵盖劳务实名制管理、智能考勤、视频监控AI识别、危大工程监控、环境监测、材料管理和进度管理等,实现工地安全、高效的智慧化管理。
|
1月前
|
消息中间件 NoSQL Java
Spring Cloud项目实战Spring Cloud视频教程 含源码
Spring Cloud项目实战Spring Cloud视频教程 含源码
33 1
|
2月前
|
设计模式 Java Spring
【Spring源码】WebSocket做推送动作的底层实例是谁
我们都知道WebSocket可以主动推送消息给用户,那做推送动作的底层实例究竟是谁?我们先整体看下整个模块的组织机构。可以看到handleMessage方法定义了每个消息格式采用不同的消息处理方法,而这些方法该类并**没有实现**,而是留给了子类去实现。
27 1
【Spring源码】WebSocket做推送动作的底层实例是谁
|
2月前
|
存储 设计模式 Java
【Spring源码】Bean采用什么数据结构进行存储
我们再来看看中间新加入的阅读线索4,不知大家忘记了没。我们可以对照图片1的代码组织结构,发现这些没存储在包里的功能类都是比较杂乱的,想必是Spring觉得目前这些功能类还构不成一个包的体系,可能后面规模更大会统一集成起来管理。
30 1
【Spring源码】Bean采用什么数据结构进行存储
|
1天前
|
监控 Java 应用服务中间件
Spring Boot 源码面试知识点
【5月更文挑战第12天】Spring Boot 是一个强大且广泛使用的框架,旨在简化 Spring 应用程序的开发过程。深入了解 Spring Boot 的源码,有助于开发者更好地使用和定制这个框架。以下是一些关键的知识点:
12 6
|
2天前
|
Java 应用服务中间件 测试技术
深入探索Spring Boot Web应用源码及实战应用
【5月更文挑战第11天】本文将详细解析Spring Boot Web应用的源码架构,并通过一个实际案例,展示如何构建一个基于Spring Boot的Web应用。本文旨在帮助读者更好地理解Spring Boot的内部工作机制,以及如何利用这些机制优化自己的Web应用开发。
11 3
|
5天前
|
存储 前端开发 Java
Spring Boot自动装配的源码学习
【4月更文挑战第8天】Spring Boot自动装配是其核心机制之一,其设计目标是在应用程序启动时,自动配置所需的各种组件,使得应用程序的开发和部署变得更加简单和高效。下面是关于Spring Boot自动装配的源码学习知识点及实战。
13 1
|
6天前
|
传感器 人工智能 前端开发
JAVA语言VUE2+Spring boot+MySQL开发的智慧校园系统源码(电子班牌可人脸识别)Saas 模式
智慧校园电子班牌,坐落于班级的门口,适合于各类型学校的场景应用,班级学校日常内容更新可由班级自行管理,也可由学校统一管理。让我们一起看看,电子班牌有哪些功能呢?
47 4
JAVA语言VUE2+Spring boot+MySQL开发的智慧校园系统源码(电子班牌可人脸识别)Saas 模式
|
13天前
|
设计模式 安全 Java
【初学者慎入】Spring源码中的16种设计模式实现
以上是威哥给大家整理了16种常见的设计模式在 Spring 源码中的运用,学习 Spring 源码成为了 Java 程序员的标配,你还知道Spring 中哪些源码中运用了设计模式,欢迎留言与威哥交流。