手写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信息,调用BeanFactory的getBean()方法,判断是否为单例模式,若为单例模式则调用SingletonBeanRegistry的getSingleton()方法,如果容器中存在Bean则直接返回,不存在则调用InstantiationStrategy的instantiate()方法,使用反射的方式生成Bean对象
在属性注入中,需要注入的属性信息封装在BeanDefinition的PropertyValues中,其本质为一个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中包含有initMethodName和destroyMethodName两个属性,用于指派Bean的初始化方法和销毁方法,可以在不实现上述接口的情况下使用反射的方式实现初始化和销毁方法,在XML中配置init-method和destroy-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进行自定义扩展开发,其包括如下几类接口
BeanFactoryAwareBeanClassLoaderAwareBeanNameAwareApplicationContextAware
事件监听机制
使用观察者模式,设置了事件监听机制
ApplicationEvent:继承自EventObject接口,Spring Event事件抽象类,后续的所有事件类都继承自该类ApplicationListener<E extends ApplicationEvent>:继承自EventListener接口,其泛型类型为该监听器关注的事件类型ApplicationEventMulticaster:事件广播器接口,定义有添加和删除监听器以及广播监听事件的方法
在使用过程中,首先调用ApplicationEventMulticaster的addApplicationListener方法将自定义监听器加入到广播器中,当要发布事件时,调用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:切入点接口,定义用于获取ClassFilter、MethodMatcher的两个类AopProxy:代理接口,用于获取代理类
使用了JDK和AspectJ两种方式实现了上述接口,并通过ProxyFactory工厂类来对两种实现的选择进行了封装
AOP融入Bean生命周期
将AOP融入Bean的声明周期,其利用了BeanPostProcessor接口,使用该接口的postProcessAfterInitialization对实例化后的原对象进行包装,返回其代理对象
其定义了Advisor访问者接口,整合切面pointcut和拦截器advice
对于反射要用的advice接口,定义了继承自该接口的BeforeAdvcice等接口,提供前置方法等AOP方法接口。同样地,对于拦截器MethodInterceptor,也封装了对应的MethodBeforeAdviceInterceptor等类,来实现各种AOP操作
相关文章:
手写spring笔记
手写spring笔记 《Spring 手撸专栏》笔记 IoC部分 Bean初始化和属性注入 Bean的信息封装在BeanDefinition中 /*** 用于记录Bean的相关信息*/ public class BeanDefinition {/*** Bean对象的类型*/private Class beanClass;/*** Bean对象中的属性信息*/private PropertyVal…...
shell的两种属性: 交互(interactive)与登录(login)
1. 背景 在看shell变量的时候引起了兴趣: 局部变量,全局变量,环境变量,shell的配置文件,参考博客: http://c.biancheng.net/view/773.html 2. 交互式与非交互式 参考博客: shell的两个属性:是否交互式(interactive), 是否登录…...
实现简单的element-table的拖拽效果
第一步,先随便创建element表格 <el-table ref"dragTable" :data"tableData" style"width: 100%" border fit highlight-current-row><el-table-column label"日期" width"180"><template slot-sc…...
Web网页浏览器远程访问jupyter notebook服务器【内网穿透】
文章目录 前言1. Python环境安装2. Jupyter 安装3. 启动Jupyter Notebook4. 远程访问4.1 安装配置cpolar内网穿透4.2 创建隧道映射本地端口 5. 固定公网地址 前言 Jupyter Notebook,它是一个交互式的数据科学和计算环境,支持多种编程语言,如…...
干翻Dubbo系列第十一篇:Dubbo常见协议与通信效率对比
文章目录 文章说明 一:协议 1:什么是协议 2:协议和序列化关系 3:协议组成 (一):头信息 (二):体信息 4:Dubbo3中常见的协议 5:…...
春秋云镜 CVE-2020-17530
春秋云镜 CVE-2020-17530 S2-061 靶标介绍 对CVE-2019-0230的绕过,Struts2官方对CVE-2019-0230的修复方式是加强OGNL表达式沙盒,而CVE-2020-17530绕过了该沙盒。当对标签属性中的原始用户输入进行评估时,强制 OGNL 评估可能会导致远程代码执…...
【java毕业设计】基于Spring Boot+Vue+mysql的论坛管理系统设计与实现(程序源码)-论坛管理系统
基于Spring BootVuemysql的论坛管理系统设计与实现(程序源码毕业论文) 大家好,今天给大家介绍基于Spring BootVuemysql的论坛管理系统设计与实现,本论文只截取部分文章重点,文章末尾附有本毕业设计完整源码及论文的获取…...
华为在ospf area 0单区域的情况下结合pbr对数据包的来回路径进行控制
配置思路: 两边去的包在R1上用mqc进行下一跳重定向 两边回程包在R4上用mqc进行下一跳重定向 最终让内网 192.168.10.0出去的数据包来回全走上面R-1-2-4 192.168.20.0出去的数据包来回全走 下面R1-3-4 R2和R3就是简单ospf配置和宣告,其它没有配置&#…...
PyQt5登录界面跳转
目录 1、设计ui界面 2、设计逻辑代码,实现登录界面跳转 3、结果 1、设计ui界面 设计后的ui界面 在这里可以设置密码不显示 这里可以设置快捷键 最后将ui界面转为py文件后获得的逻辑代码为:(文件名为Login.py) # -*- coding: u…...
git add 用法
git add 是 Git 的一个命令,用于将更改的文件加入到暂存区(staging area),准备提交这些更改。以下是该命令的常见用法: 添加单个文件 git add 文件名添加多个文件 git add 文件名1 文件名2 ...添加所有当前目录下的更改…...
系统架构设计师---2018年下午试题1分析与解答(试题三)
系统架构设计师---2018年下午试题1分析与解答 试题三 阅读以下关于嵌入式实时系统相关技术的叙述,在答题纸上回答问题 1 和问题 2。 【说明】 某公司长期从事宇航领域嵌入式实时系统的软件研制任务。公司为了适应未来嵌入式系统网络化、智能化和综合化的技术发展需要,决定…...
面试时如何回答接口测试怎么进行
一、什么是接口测试 接口测试顾名思义就是对测试系统组件间接口的一种测试,接口测试主要用于检测外部系统与系统之间以及内部各个子系统之间的交互点。测试的重点是要检查数据的交换,传递和控制管理过程,以及系统间的相互逻辑依赖关系等。 …...
【LeetCode】647.回文子串
题目 给你一个字符串 s ,请你统计并返回这个字符串中 回文子串 的数目。 回文字符串 是正着读和倒过来读一样的字符串。 子字符串 是字符串中的由连续字符组成的一个序列。 具有不同开始位置或结束位置的子串,即使是由相同的字符组成,也会…...
【Git】SSH到底是什么
一、SSH初探 1、SSH是什么? SSH是一个安全协议,类似有SSL、TSL Git有四种协议:本地协议、Git协议、HTTP协议、SSH协议 SSH协议的优缺点: 优点:SSH访问更加安全,有利于公司的开发维护,并且可…...
当你出差在外时,怎样轻松访问远程访问企业局域网象过河ERP系统?
文章目录 概述1.查看象过河服务端端口2.内网穿透3. 异地公网连接4. 固定公网地址4.1 保留一个固定TCP地址4.2 配置固定TCP地址 5. 使用固定地址连接 概述 ERP系统对于企业来说重要性不言而喻,不管是财务、生产、销售还是采购,都需要用到ERP系统来协助。…...
机器学习与模型识别1:SVM(支持向量机)
一、简介 SVM是一种二类分类模型,在特征空间中寻找间隔最大的分离超平面,使得数据得到高效的二分类。 二、SVM损失函数 SVM 的三种损失函数衡量模型的性能。 1. 0-1 损失: 当正例样本落在 y0 下方则损失为 0,否则损失为…...
在CentOS7.9上安装最新版本Docker安装步骤
1、查看系统版本: cat /etc/redhat-release 2、卸载原有版本(如有的话) sudo yum remove docker \ docker-client \ docker-client-latest \ docker-common \ d…...
基于 spring boot 的动漫信息管理系统【源码在文末】
半山腰总是最挤的,你得去山顶看看 大学生嘛,论文写不出,代码搞不懂不要紧,重要的是,从这一刻就开始学习,立刻马上! 今天带来的是最新的选题,基于 spring boot 框架的动漫信息管理系…...
vue项目根据word模版导出word文件
一、安装依赖 //1、docxtemplaternpm install docxtemplater pizzip -S//2、jszip-utilsnpm install jszip-utils -S//3、pizzipnpm install pizzip -S//4、FileSaver npm install file-saver --save二、创建word模版 也就是编辑一个word文档,文档中需要动态取值的…...
PHP 从 URL(链接) 字符串中获取参数
PHP 从 URL(链接) 字符串中获取参数 //URL(链接)字符串 $url https://www.baidu.com/?name小洪帽i&sex男&age999; //parse_url 函数从一个 URL 字符串中获取参数 $urlparse_url($url); //输出获取到的内容 echo "<pre>"; pri…...
Linux链表操作全解析
Linux C语言链表深度解析与实战技巧 一、链表基础概念与内核链表优势1.1 为什么使用链表?1.2 Linux 内核链表与用户态链表的区别 二、内核链表结构与宏解析常用宏/函数 三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势5.1 插入效率5.2 安全…...
java_网络服务相关_gateway_nacos_feign区别联系
1. spring-cloud-starter-gateway 作用:作为微服务架构的网关,统一入口,处理所有外部请求。 核心能力: 路由转发(基于路径、服务名等)过滤器(鉴权、限流、日志、Header 处理)支持负…...
【快手拥抱开源】通过快手团队开源的 KwaiCoder-AutoThink-preview 解锁大语言模型的潜力
引言: 在人工智能快速发展的浪潮中,快手Kwaipilot团队推出的 KwaiCoder-AutoThink-preview 具有里程碑意义——这是首个公开的AutoThink大语言模型(LLM)。该模型代表着该领域的重大突破,通过独特方式融合思考与非思考…...
2025盘古石杯决赛【手机取证】
前言 第三届盘古石杯国际电子数据取证大赛决赛 最后一题没有解出来,实在找不到,希望有大佬教一下我。 还有就会议时间,我感觉不是图片时间,因为在电脑看到是其他时间用老会议系统开的会。 手机取证 1、分析鸿蒙手机检材&#x…...
自然语言处理——Transformer
自然语言处理——Transformer 自注意力机制多头注意力机制Transformer 虽然循环神经网络可以对具有序列特性的数据非常有效,它能挖掘数据中的时序信息以及语义信息,但是它有一个很大的缺陷——很难并行化。 我们可以考虑用CNN来替代RNN,但是…...
.Net Framework 4/C# 关键字(非常用,持续更新...)
一、is 关键字 is 关键字用于检查对象是否于给定类型兼容,如果兼容将返回 true,如果不兼容则返回 false,在进行类型转换前,可以先使用 is 关键字判断对象是否与指定类型兼容,如果兼容才进行转换,这样的转换是安全的。 例如有:首先创建一个字符串对象,然后将字符串对象隐…...
服务器--宝塔命令
一、宝塔面板安装命令 ⚠️ 必须使用 root 用户 或 sudo 权限执行! sudo su - 1. CentOS 系统: yum install -y wget && wget -O install.sh http://download.bt.cn/install/install_6.0.sh && sh install.sh2. Ubuntu / Debian 系统…...
云原生安全实战:API网关Kong的鉴权与限流详解
🔥「炎码工坊」技术弹药已装填! 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 一、基础概念 1. API网关(API Gateway) API网关是微服务架构中的核心组件,负责统一管理所有API的流量入口。它像一座…...
通过 Ansible 在 Windows 2022 上安装 IIS Web 服务器
拓扑结构 这是一个用于通过 Ansible 部署 IIS Web 服务器的实验室拓扑。 前提条件: 在被管理的节点上安装WinRm 准备一张自签名的证书 开放防火墙入站tcp 5985 5986端口 准备自签名证书 PS C:\Users\azureuser> $cert New-SelfSignedCertificate -DnsName &…...
mac:大模型系列测试
0 MAC 前几天经过学生优惠以及国补17K入手了mac studio,然后这两天亲自测试其模型行运用能力如何,是否支持微调、推理速度等能力。下面进入正文。 1 mac 与 unsloth 按照下面的进行安装以及测试,是可以跑通文章里面的代码。训练速度也是很快的。 注意…...
