BeanDefinitionReader接口,Spring加载Bean的过程(非常流畅和容易理解)(Spring源码分析1)

本文主要是介绍BeanDefinitionReader接口,Spring加载Bean的过程(非常流畅和容易理解)(Spring源码分析1),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、前言

        前言部分,介绍Spring框架的工作和大致原理,有基础的小伙伴可以跳过

        我们现在最常使用的开发框架SSM,分别是Spring、Spring MVC和Mybatis,其功能已经超出原生Spring非常多,所以想学习Spring原理,首先要走到底层。

        第一,Spring框架,最关键的功能是bean的管理。

        bean,可以理解成一个实例化的实体对象,使用过Spring的都知道,我们配置XML文件或者注解开发时,需要配置一个bean属性【在注解中,则是使用@Component注解】

        这个配置的目的,就是告诉Spring某些信息。

        最重要也是最基础的两个,一个是bean指向的类,另一个是bean的名称。

        当然,还有单例或原型、初始化方法、销毁方法等等。

        不过所有的这些信息,都指向BeanDefinition

        BeanDefinition,就是Spring对bean的定义,我们要使用bean,绕不开这个接口。

        第二,Spring的核心设计,在于如何用配置文件,管理Bean

        学习过设计模式的同学,应该对“工厂模式”有所了解,在此不介绍基础原理。

        我们只需记住,Spring中,工厂是用BeanFactory接口体现的

        不过我们常用的工厂方法,很明显有一个缺点:对于接口Data,它每增加一个实现类,就要配置相关信息。

        这也是Spring的问题所在,所以我们一旦增加一个类,就要用@Component注解该类。

        我们考虑一个问题:明明只有配置文件,为何Spring框架却能将XML文件中的Bean定义,实例化出来呢?

        记住下面的流程

        第一步,必须把Bean信息读取出来,也就是说,要有读取XML文件的能力。

        第二步,Spring框架必须能实例化这个Bean,并且要有至少一个容器【数据结构】,来存储这些实例对象。

        第三步,Spring至少对外有一个接口,使外界可以得到这个Bean。

二、从代码入手,学习底层的Spring使用方法【下面会统一说明这些名词】     

        我们考虑下面这行代码,现在已经很少有人这么写了,不过为了学习Spring的原理,我们必须先把Spring的Context部分放在一旁,专注于beans这个核心部分。

    public static void main(String[] args) throws Exception{// 建造Spring工厂DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory();// 初始化读取器。XmlBeanDefinitionReader beanDefinitionReader =new XmlBeanDefinitionReader(beanFactory);// 调用 加载 方法。【注:虽然是叫“加载”方法,但事实上完成了加载、注册和实例化功能】beanDefinitionReader.loadBeanDefinitions("Spring.xml");MyService myService = (MyService) beanFactory.getBean("myService");myService.serve(); // 只有一个打印方法}

        这段代码做了什么事呢?

        按流程来说,请保持疑问。

        第一,new一个BeanFactory工厂

                疑问1:为什么使用Default Listable Bean Factory,而不是使用BeanFactory作为声明?

                疑问2:为什么这个对象的new,不需要参数?那把它new出来有什么用处?

        第二,new一个BeanDefinitionReader读取器

                疑问1:Xml BeanDefinitionReader和我说的BeanDefinitionReader有什么关系?

                疑问2:将这个BeanFactory传递进去,会发生什么?为什么需要这个Factory?

        第三,调用读取器reader的load方法

                疑问1:这个方法做了什么?为什么一下子就能使用Bean了?

三、记住几个重点名词

        首先,我要说一下几个重点名词,以及它的含义。

        记住:loadBeanDefinitions方法、Resource接口、resourceLoader属性。

        按照类来解释。

        BeanDefinition

                Bean的定义,这个我想大家都了解,不过请记住一点:BeanDefinition是Spring的唯一数据类,这个类定义了所有数据,其它类定义的数据,都是为了辅助这个类的使用。

        BeanFactory

                工厂类,但不负责加载、初始化Bean,这个接口的唯一作用,其实是管理Bean【比如在外界需要时,传递一个bean对象】

                这个接口的下层实现接口非常多,比如ListableBeanFactory,在此不介绍。

                最重要的一个实现类,即DefaultListableBeanFactory,这个类同时实现了另一条支线上的BeanDefinitionRegistry接口。

                这个实现类,有一个HashMap的数据结构beanDefinitionMap,存储了所有的bean,无论是bean的注册还是初始化,都要使用到它。

                这就相当于生出了一个同时拥有二者能力的儿子。

        BeanDefinitionRegistry

                一个拥有注册Bean能力的接口,并非本文重点。

                不过在Bean加载结束后,注册就要依靠它。

        BeanDefinitionReader

                这是本文的重点之一,这个接口的类继承结构如下图所示。

                我们忽略Groovy、Properties,因为它们不是很重要。

                Reader接口只定义了一些方法,在此不赘叙。

                在Abstract中,实现了通用的方法,其中最重要的就是loadBeanDefinitions方法,这个方法的重载很多,实际的加载过程,甚至放在子类中实现。

                不过不影响它的重要性,因为它将资源文件,全部转化为Resource资源,以便可以统一地访问数据。

                至此,介绍完毕。

四、加载Bean,Spring的初始工作

        有了前文的基础,这一步就轻松很多,让我们来看看源码。

        我必须解释一下,BeanDefinitionReader读取器,它的new需要一个factory的原因,是它在加载方法中,调用了Bean注册、初始化的方法。

        如果单纯的“加载”资源,使其成为可读的Document格式,不需要一个factory。【当然,实际上这个factory在reader眼里,是一个rigistry注册器】

        第一步,走进Abstract抽象类loadBeanDefinitions方法里【参数:String location】

                为了使代码结构更清晰,我把异常处理、日志记录都删除了,以下是我的分析。【忽略如何拿到加载器、以及如何加载资源】

                我再次强调一下,Resource类,是Spring对底层资源的统一访问接口,所以拿到Resource对象是第一步。

                首先,拿到资源加载器。

                然后,根据加载器的类型【一次读取n个资源,或者1个资源】,拿到资源。

                最后,把资源放入集合,等待处理。

public int loadBeanDefinitions(String location, @Nullable Set<Resource> actualResources){ResourceLoader resourceLoader = this.getResourceLoader();  int count;// 加载一个或多个资源if (resourceLoader instanceof ResourcePatternResolver) {// 多资源加载器,一次加载多个资源Resource[] resources = ((ResourcePatternResolver)resourceLoader).getResources(location);count = this.loadBeanDefinitions(resources);if (actualResources != null) {// 将资源类Resource,放进集合Collections.addAll(actualResources, resources);}return count;    }} else {// 单资源加载器,一次加载一个资源Resource resource = resourceLoader.getResource(location);count = this.loadBeanDefinitions((Resource)resource);if (actualResources != null) {actualResources.add(resource);}return count;}}}

        第二步,走进Xml子类实现的loadBeanDefinitions方法中【参数:Resource resource】

                这一步骤里,我依旧删除了异常处理、日志和一些语句,保留了比较重要的方法。

                EncodedResource相当于Resource的编码结果,与Resource差别不大,在此忽略。

                主要做了两件事

                第一,将资源转化为InputSource。

                第二,调用doLoadBeanDefinitions方法。【真正的加载bean方法】

public int loadBeanDefinitions(EncodedResource encodedResource){Assert.notNull(encodedResource, "EncodedResource must not be null");Set<EncodedResource> currentResources = (Set)this.resourcesCurrentlyBeingLoaded.get();int var6;InputStream inputStream = encodedResource.getResource().getInputStream();Throwable var4 = null;// 将资源转化为InputSourceInputSource inputSource = new InputSource(inputStream);if (encodedResource.getEncoding() != null) {inputSource.setEncoding(encodedResource.getEncoding());}// 最重要的一步var6 = this.doLoadBeanDefinitions(inputSource, encodedResource.getResource());return var6;
}

        第三步,走进doLoadBeanDefinitions方法

                这一步的主要逻辑,就是得到Document文件,然后调用registerBeanDefinitions方法。

Document doc = this.doLoadDocument(inputSource, resource);
int count = this.registerBeanDefinitions(doc, resource);return count;

        至此,Bean的加载已经完毕,Document类,已经可以支持Spring注册bean和实例化bean,从底层XML资源,转化为可读的资源,已经完成。

        至于Resource、Document是什么,ResourceLoader的原理又是什么,这部分我在下次再写。

五、结语

以上就是我分享的对Spring框架的理解。

         我是蚊子码农,如有补充或者疑问,欢迎在评论区留言。个人的知识体系可能没有那么完善,希望各位多多指正,谢谢大家。

这篇关于BeanDefinitionReader接口,Spring加载Bean的过程(非常流畅和容易理解)(Spring源码分析1)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1031834

相关文章

解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题

《解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题》本文主要讲述了在使用MyBatis和MyBatis-Plus时遇到的绑定异常... 目录myBATis-plus-boot-starpythonter与mybatis-spring-b

Java中switch-case结构的使用方法举例详解

《Java中switch-case结构的使用方法举例详解》:本文主要介绍Java中switch-case结构使用的相关资料,switch-case结构是Java中处理多个分支条件的一种有效方式,它... 目录前言一、switch-case结构的基本语法二、使用示例三、注意事项四、总结前言对于Java初学者

关于Java内存访问重排序的研究

《关于Java内存访问重排序的研究》文章主要介绍了重排序现象及其在多线程编程中的影响,包括内存可见性问题和Java内存模型中对重排序的规则... 目录什么是重排序重排序图解重排序实验as-if-serial语义内存访问重排序与内存可见性内存访问重排序与Java内存模型重排序示意表内存屏障内存屏障示意表Int

最好用的WPF加载动画功能

《最好用的WPF加载动画功能》当开发应用程序时,提供良好的用户体验(UX)是至关重要的,加载动画作为一种有效的沟通工具,它不仅能告知用户系统正在工作,还能够通过视觉上的吸引力来增强整体用户体验,本文给... 目录前言需求分析高级用法综合案例总结最后前言当开发应用程序时,提供良好的用户体验(UX)是至关重要

Java汇编源码如何查看环境搭建

《Java汇编源码如何查看环境搭建》:本文主要介绍如何在IntelliJIDEA开发环境中搭建字节码和汇编环境,以便更好地进行代码调优和JVM学习,首先,介绍了如何配置IntelliJIDEA以方... 目录一、简介二、在IDEA开发环境中搭建汇编环境2.1 在IDEA中搭建字节码查看环境2.1.1 搭建步

Redis主从复制实现原理分析

《Redis主从复制实现原理分析》Redis主从复制通过Sync和CommandPropagate阶段实现数据同步,2.8版本后引入Psync指令,根据复制偏移量进行全量或部分同步,优化了数据传输效率... 目录Redis主DodMIK从复制实现原理实现原理Psync: 2.8版本后总结Redis主从复制实

javafx 如何将项目打包为 Windows 的可执行文件exe

《javafx如何将项目打包为Windows的可执行文件exe》文章介绍了三种将JavaFX项目打包为.exe文件的方法:方法1使用jpackage(适用于JDK14及以上版本),方法2使用La... 目录方法 1:使用 jpackage(适用于 JDK 14 及更高版本)方法 2:使用 Launch4j(

JAVA利用顺序表实现“杨辉三角”的思路及代码示例

《JAVA利用顺序表实现“杨辉三角”的思路及代码示例》杨辉三角形是中国古代数学的杰出研究成果之一,是我国北宋数学家贾宪于1050年首先发现并使用的,:本文主要介绍JAVA利用顺序表实现杨辉三角的思... 目录一:“杨辉三角”题目链接二:题解代码:三:题解思路:总结一:“杨辉三角”题目链接题目链接:点击这里

SpringBoot使用注解集成Redis缓存的示例代码

《SpringBoot使用注解集成Redis缓存的示例代码》:本文主要介绍在SpringBoot中使用注解集成Redis缓存的步骤,包括添加依赖、创建相关配置类、需要缓存数据的类(Tes... 目录一、创建 Caching 配置类二、创建需要缓存数据的类三、测试方法Spring Boot 熟悉后,集成一个外

使用JavaScript操作本地存储

《使用JavaScript操作本地存储》这篇文章主要为大家详细介绍了JavaScript中操作本地存储的相关知识,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... 目录本地存储:localStorage 和 sessionStorage基本使用方法1. localStorage