手写spring笔记

手写spring笔记

《Spring 手撸专栏》笔记

IoC部分

Bean初始化和属性注入

Bean的信息封装在BeanDefinition

/*** 用于记录Bean的相关信息*/
public class BeanDefinition {/*** Bean对象的类型*/private Class beanClass;/*** Bean对象中的属性信息*/private PropertyValues propertyValues;/*** 初始化方法的名称*/private String initMethodName;/*** 销毁方法的名称*/private String destroyMethodName;/*** 默认为单例模式*/private String scope;
}

Bean注入过程中有一下几个主要接口:

  • BeanFactory:Bean工厂接口,其中声明了获取Bean对象的方法
  • SingletonBeanRegistry: Bean的单例模式创建接口,声明了单例对象的创建方法和销毁方法
  • InstantiationStrategy:实例化Bean对象的策略接口,申明了Bean对象使用何种方式实例化对象的方法,在实现类中使用反射的方式获取构造方法来得到Bean对象,有JDK和Cglib两种实现
  • BeanDefinitionRegistry:管理BeanDefinition的注册,其实现类中包含存有BeanDefinition的Map

其具体流程为:
BeanDefinitionRegistry得到BeanDefinition信息,调用BeanFactorygetBean()方法,判断是否为单例模式,若为单例模式则调用SingletonBeanRegistrygetSingleton()方法,如果容器中存在Bean则直接返回,不存在则调用InstantiationStrategyinstantiate()方法,使用反射的方式生成Bean对象

在属性注入中,需要注入的属性信息封装在BeanDefinitionPropertyValues中,其本质为一个PropertyValue列表。

在创建Bean对象时,若存在无参构造方法,则使用无参构造方法,若没有,则从PropertyValues去除构造方法需要的属性。

/*** Bean对象中的属性值*/
public class PropertyValue {/*** Bean对象属性的名称*/private String name;/*** Bean对象中属性的实例化对象*/private Object value;
}

资源加载器

Spring需要解析配置文件,对此,定义了一下接口:

  • Resource:资源信息接,用于处理资源加载流,其实现类根据配置文件地址得到资源信息,并向外声明有得到输入流的接口,其包含XML文件配置和URL文件配置等方式
  • ResourceLoader:资源加载器接口,通过该接口获取路径对应的资源对象,用于获得资源对象
  • BeanDefinitionReader:BeanDefinition读取接口,用于读取配置文件并调用BeanDefinitionRegistry加载BeanDefinition

Bean生命周期

Bean的初始化操作,提供了Bean的初始化和销毁等方法接口,其包括:

  • InitializingBean:提供了初始化方法的Bean,如果Bean实现了该接口,则会在创建Bean时调用初始化方法
  • DisposableBean:提供了Bean销毁方法,若实现了该接口,则在容器销毁时调用其销毁方法

此外,在BeanDefinition中包含有initMethodNamedestroyMethodName两个属性,用于指派Bean的初始化方法和销毁方法,可以在不实现上述接口的情况下使用反射的方式实现初始化和销毁方法,在XML中配置init-methoddestroy-method两个属性即可

同时,定义有一下两个接口,来管理Bean的初始化操作

/*** Bean实例化前对其进行预处理的接口,提供修改BeanDefinition的方法*/
public interface BeanFactoryPostProcessor {/*** 所有的BeanDefinition加载完成后而Bean对象实例化之前调用,提供修改BeanDefinition的机制** @param beanFactory* @throws BeansException*/void postProcessorBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException;
}
/*** 对Bean对象初始化前后进行处理的接口,提供Bean初始化前后对其进行操作的方法*/
public interface BeanPostProcessor {/*** 在Bean对象执行初始化前对Bean实例对象进行操作** @param bean     被操作的Bean对象* @param beanName 被操作的Bean对象的名称* @return* @throws BeansException*/Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException;/*** 在Bean对象执行初始化后对Bean实例对象进行操作** @param bean     被操作的Bean对象* @param beanName 被操作的Bean对象的名称* @return* @throws BeansException*/Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException;
}

Bean工厂接口

对于复杂类型的Bean,其提供了FactoryBean<T>接口,对于实现了该接口的类,spring在创建Bean对象时,不是直接生成该对象,而是调用getObject方法来生成Bean对象

/*** Bean对象的工厂接口,声明了从工厂获得Bean对象的方法*/
public interface BeanFactory {/*** 有参方式获取Bean** @param name Bean名称* @return Bean对象实例* @throws BeansException Bean创建时异常*/Object getBean(String name) throws BeansException;/*** 有参方式获取Bean** @param name Bean名称* @param args 参数* @return Bean对象实例* @throws BeansException Bean创建时异常*/Object getBean(String name, Object... args) throws BeansException;/*** 按类型获取Bean对象** @param name         Bean对象名称* @param requiredType Bean对象类型* @param <T>          Bean对象类型* @return Bean对象实例*/<T> T getBean(String name, Class<T> requiredType) throws BeansException;<T> T getBean(Class<T> requiredType) throws BeansException;
}

Aware感知接口

对于实现了Aware接口的Bean,在spring创建实例时,会将对应的信息注入到Bean中,方便使用者对spring进行自定义扩展开发,其包括如下几类接口

  • BeanFactoryAware
  • BeanClassLoaderAware
  • BeanNameAware
  • ApplicationContextAware

事件监听机制

使用观察者模式,设置了事件监听机制

  • ApplicationEvent:继承自EventObject接口,Spring Event事件抽象类,后续的所有事件类都继承自该类
  • ApplicationListener<E extends ApplicationEvent>:继承自EventListener接口,其泛型类型为该监听器关注的事件类型
  • ApplicationEventMulticaster:事件广播器接口,定义有添加和删除监听器以及广播监听事件的方法

在使用过程中,首先调用ApplicationEventMulticasteraddApplicationListener方法将自定义监听器加入到广播器中,当要发布事件时,调用multicastEvent方法,该方法会便利已注册的所有监听器,并向关注该事件的监听器发送通知,调用其onApplicationEvent方法

应用上下文

接口ApplicationContext整合了上述的各种方法,并提供了向外的操作接口。其中,ApplicationContext的实现类中定义了核心方法refresh(),其具体内容如下:

    @Overridepublic void refresh() throws BeansException {// 创建BeanFactory,并加载BeanDefinitionrefreshBeanFactory();// 获取BeanFactoryConfigurableListableBeanFactory beanFactory = getBeanFactory();// 添加ApplicationContextAwareProcessor对象处理ApplicationContextAwarebeanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this));// 在Bean实例化之前,执行BeanFactoryPostProcessor方法invokeBeanFactoryPostProcessors(beanFactory);// 将BeanPostProcessor提前注册到容器中registerBeanPostProcessors(beanFactory);// 提前实例化单例Bean对象beanFactory.preInstantiateSingletons();// 初始化事件发布者initApplicationEventMulticaster();// 注册事件监听器registerListeners();// 发布容器刷新完成事件finishRefresh();}

三级缓存机制

为了解决循环依赖,spring使用了三级缓存,如下所示:

    /*** Bean容器一级缓存,用于存储Bean的完整实例化对象*/private Map<String, Object> singletonObjects = new HashMap<>();/*** Bean容器二级缓存,用于存储Bean的早期非完整实例化对象*/private Map<String, Object> earlySingletonObjects = new HashMap<>();/*** Bean容器三级缓存,用于存储Bean的代理对象*/private Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>();/**

其中,三级缓存存储的不是Bean对象,而是一个工厂对象,其目的是为了提早暴露Bean,等到后续的Bean依赖该Bean的时候,就会调用工厂的getObject方法获取到Bean对象。三级缓存的顺序如下:

    @Overridepublic Object getSingleton(String beanName) throws BeansException {Object singletonObject = singletonObjects.get(beanName);if (singletonObject == null) {// 如果没有完整对象,则从二级缓存中获取未完整实例化的对象singletonObject = earlySingletonObjects.get(beanName);if (singletonObject == null) {// 如果二级缓存中也没有,则从代理对象处生成未完成的实例化对象,并删除代理对象ObjectFactory<?> singletonFactory = singletonFactories.get(beanName);if (singletonFactory != null) {singletonObject = singletonFactory.getObject();earlySingletonObjects.put(beanName, singletonObject);singletonFactories.remove(beanName);}}}return singletonObject;}

注解配置信息

这里给出了一下几个注解功能的实现:

  • @Component:将类注入到容器中
  • @Autowired:对Bean对象对应的该属性进行依赖注入
  • @Scope:配置该类的作用域为单例还是原型
  • @Value:注入配置文件中的信息

通过读取配置文件中<context:component-scan />指定的路径,对其下所有的类进行扫描,若标注有@Component注解,则生成对应的BeanDefinition,并读取其属性,生成对应的PropertyValue,若属性标注有@Value注解,表示属性值为配置文件中的对应数值,若标注有@Autowired注解,则表示属性值为Bean。生成好BeanDefinition后,就和原本流程一样了

AOP

切点和匹配接口定义

AOP的核心是使用代理的方式生成代理对象,其核心接口包括:

  • ClassFilter:定义类匹配类,用于切点找到给定的接口和目标类,提供了判断切入点是否应用在给定的接口或目标类中的方法
  • MethodMatcher:方法匹配类,找到表达式范围内匹配下的目标类和方法,提供了判断给定的方法是否匹配表达式的方法
  • Pointcut:切入点接口,定义用于获取ClassFilterMethodMatcher的两个类
  • AopProxy:代理接口,用于获取代理类

使用了JDK和AspectJ两种方式实现了上述接口,并通过ProxyFactory工厂类来对两种实现的选择进行了封装

AOP融入Bean生命周期

将AOP融入Bean的声明周期,其利用了BeanPostProcessor接口,使用该接口的postProcessAfterInitialization对实例化后的原对象进行包装,返回其代理对象

其定义了Advisor访问者接口,整合切面pointcut和拦截器advice

对于反射要用的advice接口,定义了继承自该接口的BeforeAdvcice等接口,提供前置方法等AOP方法接口。同样地,对于拦截器MethodInterceptor,也封装了对应的MethodBeforeAdviceInterceptor等类,来实现各种AOP操作

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/41021.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MFC第三十天 通过CToolBar类开发文字工具栏和工具箱、GDI+边框填充以及基本图形的绘制方法、图形绘制过程的反色线模型和实色模型

文章目录 CControlBar通过CToolBar类开发文字工具栏和工具箱CMainFrame.hCAppCMainFrm.cppCMainView.hCMainView.cppCEllipse.hCEllipse.cppCLine.hCLine.cppCRRect .hCRRect .cpp CControlBar class AFX_NOVTABLE CControlBar : public CWnd{DECLARE_DYNAMIC(CControlBar)pro…

OC调用Swift编写的framework

一、前言 随着swift趋向稳定&#xff0c;越来越多的公司都开始用swift来编写苹果相关的业务了&#xff0c;关于swift的利弊这里就不多说了。这里详细介绍OC调用swift编写的framework库的步骤 二、制作framework 1、新建项目&#xff0c;选择framework 2、填写framework的名称…

AutoHotkey:定时删除目录下指定分钟以前的文件,带UI界面

删除指定目录下&#xff0c;所有在某个指定分钟以前的文件&#xff0c;可以用来清理经常生成很多文件的目录&#xff0c;但又需要保留最新的一部分文件 支持拖放目录到界面 能够记忆设置&#xff0c;下次启动后不用重新设置&#xff0c;可以直接开始 应用场景比如&#xff1a…

WinForm内嵌Unity3D

Unity3D可以C#脚本进行开&#xff0c;使用vstu2013.msi插件&#xff0c;可以实现在VS2013中的调试。在开发完成后&#xff0c;由于项目需要&#xff0c;需要将Unity3D嵌入到WinForm中。WinForm中的UnityWebPlayer Control可以载入Unity3D。先看效果图。 一、为了能够动态设置ax…

【boost网络库从青铜到王者】第五篇:asio网络编程中的同步读写的客户端和服务器示例

文章目录 1、简介2、客户端设计3、服务器设计3.1、session函数3.2、StartListen函数3、总体设计 4、效果测试5、遇到的问题5.1、服务器遇到的问题5.1.1、不用显示调用bind绑定和listen监听函数5.1.2、出现 Error occured!Error code : 10009 .Message: 提供的文件句柄无效。 [s…

Failed to execute goal org.apache.maven.plugins

原因&#xff1a; 这个文件D:\java\maven\com\ruoyi\pg-student\maven-metadata-local.xml出了问题 解决&#xff1a; 最简单的直接删除D:\java\maven\com\ruoyi\pg-student\maven-metadata-local.xml重新打包 或者把D:\java\maven\com\ruoyi\pg-student这个目录下所有文件…

性能测试场景设计

性能测试场景设计&#xff0c;是性能测试中的重要概念&#xff0c;性能测试场景设计&#xff0c;目的是要描述如何执行性能测试。 通常来讲&#xff0c;性能测试场景设计主要会涉及以下部分&#xff1a; 并发用户数是多少&#xff1f; 测试刚开始时&#xff0c;以什么样的速率…

Spring WebFlux 的详细介绍

Spring WebFlux 是基于响应式编程的框架&#xff0c;用于构建异步、非阻塞的 Web 应用程序。它是Spring框架的一部分&#xff0c;专注于支持响应式编程范式&#xff0c;使应用程序能够高效地处理大量的并发请求和事件。 以下是关于 Spring WebFlux 的详细介绍&#xff1a; 1. *…

go入门实践五-实现一个https服务

文章目录 前言生成证书申请免费的证书使用Go语言生成自签CA证书 https的客户端和服务端服务端代码客户端代码 tls的客户端和服务端服务端客户端 前言 在公网中&#xff0c;我想加密传输的数据。(1)很自然&#xff0c;我想到了把数据放到http的请求中&#xff0c;然后通过tls确…

2023年京东宠物食品行业数据分析(京东大数据)

宠物食品市场需求主要来自于养宠规模&#xff0c;近年来由于我国宠物数量及养宠人群的规模均在不断扩大&#xff0c;宠物相关产业和市场规模也在蓬勃发展&#xff0c;宠物食品市场也同样保持正向增长。 根据鲸参谋电商数据分析平台的相关数据显示&#xff0c;2023年1月-7月&am…

vue5种模糊查询方式

在Vue中&#xff0c;有多种方式可以实现模糊查询。以下是五种常见的模糊查询方式&#xff1a; 使用JavaScript的filter()方法&#xff1a;使用filter()方法可以对数组进行筛选&#xff0c;根据指定的条件进行模糊查询。例如&#xff1a; data() {return {items: [{ name: App…

接口自动化测试(添加课程接口调试,调试合同上传接口,合同列表查询接口,批量执行)

1、我们把信息截取一下 1.1 添加一个新的请求 1.2 对整个请求进行保存&#xff0c;Ctrl S 2、这一次我们添加的是课程添加接口&#xff0c;以后一个接口完成&#xff0c;之后Ctrl S 就能够保存 2.1 选择方法 2.2 设置请求头&#xff0c;参数数据后期我们通过配置设置就行 3、…

收银一体化-亿发2023智慧门店新零售营销策略,实现全渠道运营

伴随着互联网电商行业的兴起&#xff0c;以及用户理念的改变&#xff0c;大量用户从线下涌入线上&#xff0c;传统的线下门店人流量急剧收缩&#xff0c;门店升级几乎成为了每一个零售企业的发展之路。智慧门店新零售收银解决方案是针对传统零售企业面临的诸多挑战和问题&#…

Mathematica 与 Matlab 常见复杂指令集汇编

Mathematica 常见指令汇编 Mathematica 常见指令 NDSolve 求解结果的保存 sol NDSolve[{y[x] x^2, y[0] 0, g[x] -y[x]^2, g[0] 1}, {y, g}, {x, 0, 1}]; numericSoly sol[[1, 1, 2]]; numericSolg sol[[1, 2, 2]]; data Table[{x, numericSoly[x], numericSolg[x]},…

JVM——类加载器

回顾一下类加载过程 类加载过程&#xff1a;加载->连接->初始化。连接过程又可分为三步:验证->准备->解析。 一个非数组类的加载阶段&#xff08;加载阶段获取类的二进制字节流的动作&#xff09;是可控性最强的阶段&#xff0c;这一步我们可以去完成还可以自定义…

【计算机网络篇】UDP协议

✅作者简介&#xff1a;大家好&#xff0c;我是小杨 &#x1f4c3;个人主页&#xff1a;「小杨」的csdn博客 &#x1f433;希望大家多多支持&#x1f970;一起进步呀&#xff01; UDP协议 1&#xff0c;UDP 简介 UDP&#xff08;User Datagram Protocol&#xff09;是一种无连…

Flink学习笔记(一)

流处理 批处理应用于有界数据流的处理&#xff0c;流处理则应用于无界数据流的处理。 有界数据流&#xff1a;输入数据有明确的开始和结束。 无界数据流&#xff1a;输入数据没有明确的开始和结束&#xff0c;或者说数据是无限的&#xff0c;数据通常会随着时间变化而更新。 在…

Kaptcha的基本应用

Kaptcha Kaptcha 是一个用于生成和验证验证码的 Java 库&#xff0c;提供了丰富的生成和验证功能&#xff0c;并支持自定义配置。它可以用于增加应用程序的安全性&#xff0c;防止机器人和恶意攻击。 Kaptcha 可以生成各种类型的验证码&#xff0c;包括数字、字母、数字字母组…

KDD 2023 获奖论文公布,港中文、港科大等获最佳论文奖

ACM SIGKDD&#xff08;国际数据挖掘与知识发现大会&#xff0c;KDD&#xff09;是数据挖掘领域历史最悠久、规模最大的国际顶级学术会议&#xff0c;也是首个引入大数据、数据科学、预测分析、众包等概念的会议。 今年&#xff0c;第29届 KDD 大会于上周在美国加州长滩圆满结…

HTTP--Request详解

请求消息数据格式 请求行 请求方式 请求url 请求协议/版本 GET /login.html HTTP/1.1 请求头 客户端浏览器告诉服务器一些信息 请求头名称: 请求头值 常见的请求头&#xff1a; User-Agent&#xff1a;浏览器告诉服务器&#xff0c;我访问你使用的浏览器版本信息 可…