• Spring框架系列(6) - Spring IOC实现原理详解之IOC体系结构设计


    在对IoC有了初步的认知后,我们开始对IOC的实现原理进行深入理解。本文将帮助你站在设计者的角度去看IOC最顶层的结构设计。@pdai

    站在设计者的角度考虑设计IOC容器

    如果让你来设计一个IoC容器,你会怎么设计?我们初步的通过这个问题,来帮助我们更好的理解IOC的设计。

    在设计时,首先需要考虑的是IOC容器的功能(输入和输出), 承接前面的文章,我们初步的画出IOC容器的整体功能。

    在此基础上,我们初步的去思考,如果作为一个IOC容器的设计者,主体上应该包含哪几个部分:

    • 加载Bean的配置(比如xml配置)
      • 比如不同类型资源的加载,解析成生成统一Bean的定义
    • 根据Bean的定义加载生成Bean的实例,并放置在Bean容器中
      • 比如Bean的依赖注入,Bean的嵌套,Bean存放(缓存)等
    • 除了基础Bean外,还有常规针对企业级业务的特别Bean
      • 比如国际化Message,事件Event等生成特殊的类结构去支撑
    • 对容器中的Bean提供统一的管理和调用
      • 比如用工厂模式管理,提供方法根据名字/类的类型等从容器中获取Bean

    (pdai:这种思考的过程才是建设性的,知识体系的构建才是高效的)

    Spring IoC的体系结构设计

    那么我们来看下Spring设计者是如何设计IoC并解决这些问题的。

    BeanFactory和BeanRegistry:IOC容器功能规范和Bean的注册

    Spring Bean的创建是典型的工厂模式,这一系列的Bean工厂,也即IOC容器为开发者管理对象间的依赖关系提供了很多便利和基础服务,在Spring中有许多的IOC容器的实现供用户选择和使用,这是IOC容器的基础;在顶层的结构设计主要围绕着BeanFactory和xxxRegistry进行:

    • BeanFactory: 工厂模式定义了IOC容器的基本功能规范
    • BeanRegistry: 向IOC容器手工注册 BeanDefinition 对象的方法

    其相互关系如下:

    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fulDoSKF-1656290495963)

    我们再通过几个问题来辅助理解。

    BeanFactory定义了IOC 容器基本功能规范?

    BeanFactory作为最顶层的一个接口类,它定义了IOC容器的基本功能规范,BeanFactory 有三个子类:ListableBeanFactory、HierarchicalBeanFactory 和AutowireCapableBeanFactory。我们看下BeanFactory接口:

    public interface BeanFactory {    
          
        //用于取消引用实例并将其与FactoryBean创建的bean区分开来。例如,如果命名的bean是FactoryBean,则获取将返回Factory,而不是Factory返回的实例。
        String FACTORY_BEAN_PREFIX = "&"; 
            
        //根据bean的名字和Class类型等来得到bean实例    
        Object getBean(String name) throws BeansException;    
        Object getBean(String name, Class requiredType) throws BeansException;    
        Object getBean(String name, Object... args) throws BeansException;
        <T> T getBean(Class<T> requiredType) throws BeansException;
        <T> T getBean(Class<T> requiredType, Object... args) throws BeansException;
    
        //返回指定bean的Provider
        <T> ObjectProvider<T> getBeanProvider(Class<T> requiredType);
        <T> ObjectProvider<T> getBeanProvider(ResolvableType requiredType);
    
        //检查工厂中是否包含给定name的bean,或者外部注册的bean
        boolean containsBean(String name);
    
        //检查所给定name的bean是否为单例/原型
        boolean isSingleton(String name) throws NoSuchBeanDefinitionException;
        boolean isPrototype(String name) throws NoSuchBeanDefinitionException;
    
        //判断所给name的类型与type是否匹配
        boolean isTypeMatch(String name, ResolvableType typeToMatch) throws NoSuchBeanDefinitionException;
        boolean isTypeMatch(String name, Class<?> typeToMatch) throws NoSuchBeanDefinitionException;
    
        //获取给定name的bean的类型
        @Nullable
        Class<?> getType(String name) throws NoSuchBeanDefinitionException;
    
        //返回给定name的bean的别名
        String[] getAliases(String name);
         
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35

    BeanFactory为何要定义这么多层次的接口?定义了哪些接口?

    主要是为了区分在 Spring 内部在操作过程中对象的传递和转化过程中,对对象的数据访问所做的限制

    有哪些接口呢?

    • ListableBeanFactory:该接口定义了访问容器中 Bean 基本信息的若干方法,如查看Bean 的个数、获取某一类型 Bean 的配置名、查看容器中是否包括某一 Bean 等方法;
    • HierarchicalBeanFactory:父子级联 IoC 容器的接口,子容器可以通过接口方法访问父容器; 通过 HierarchicalBeanFactory 接口, Spring 的 IoC 容器可以建立父子层级关联的容器体系,子容器可以访问父容器中的 Bean,但父容器不能访问子容器的 Bean。Spring 使用父子容器实现了很多功能,比如在 Spring MVC 中,展现层 Bean 位于一个子容器中,而业务层和持久层的 Bean 位于父容器中。这样,展现层 Bean 就可以引用业务层和持久层的 Bean,而业务层和持久层的 Bean 则看不到展现层的 Bean。
    • ConfigurableBeanFactory:是一个重要的接口,增强了 IoC 容器的可定制性,它定义了设置类装载器、属性编辑器、容器初始化后置处理器等方法;
    • ConfigurableListableBeanFactory: ListableBeanFactory 和 ConfigurableBeanFactory的融合;
    • AutowireCapableBeanFactory:定义了将容器中的 Bean 按某种规则(如按名字匹配、按类型匹配等)进行自动装配的方法;

    如何将Bean注册到BeanFactory中?BeanRegistry

    Spring 配置文件中每一个<bean>节点元素在 Spring 容器里都通过一个 BeanDefinition 对象表示,它描述了 Bean 的配置信息。而 BeanDefinitionRegistry 接口提供了向容器手工注册 BeanDefinition 对象的方法。

    BeanDefinition:各种Bean对象及其相互的关系

    Bean对象存在依赖嵌套等关系,所以设计者设计了BeanDefinition,它用来对Bean对象及关系定义;我们在理解时只需要抓住如下三个要点:

    • BeanDefinition 定义了各种Bean对象及其相互的关系
    • BeanDefinitionReader 这是BeanDefinition的解析器
    • BeanDefinitionHolder 这是BeanDefination的包装类,用来存储BeanDefinition,name以及aliases等。
    • BeanDefinition

    SpringIOC容器管理了我们定义的各种Bean对象及其相互的关系,Bean对象在Spring实现中是以BeanDefinition来描述的,其继承体系如下

    • BeanDefinitionReader

    Bean 的解析过程非常复杂,功能被分的很细,因为这里需要被扩展的地方很多,必须保证有足够的灵活性,以应对可能的变化。Bean 的解析主要就是对 Spring 配置文件的解析。这个解析过程主要通过下图中的类完成:

    • BeanDefinitionHolder

    BeanDefinitionHolder 这是BeanDefination的包装类,用来存储BeanDefinition,name以及aliases等

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-pLbx6gRT-1656290495964)(https://pdai.tech/_images/spring/springframework/spring-framework-ioc-source-4.png)]

    ApplicationContext:IOC接口设计和实现

    IoC容器的接口类是ApplicationContext,很显然它必然继承BeanFactory对Bean规范(最基本的ioc容器的实现)进行定义。而ApplicationContext表示的是应用的上下文,除了对Bean的管理外,还至少应该包含了

    • 访问资源: 对不同方式的Bean配置(即资源)进行加载。(实现ResourcePatternResolver接口)
    • 国际化: 支持信息源,可以实现国际化。(实现MessageSource接口)
    • 应用事件: 支持应用事件。(实现ApplicationEventPublisher接口)

    ApplicationContext接口的设计

    我们来看下ApplicationContext整体结构

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-kic8j3gp-1656290495965)(https://pdai.tech/_images/spring/springframework/spring-framework-ioc-source-51.png)]

    • HierarchicalBeanFactory 和 ListableBeanFactory: ApplicationContext 继承了 HierarchicalBeanFactory 和 ListableBeanFactory 接口,在此基础上,还通过多个其他的接口扩展了 BeanFactory 的功能:
    • ApplicationEventPublisher:让容器拥有发布应用上下文事件的功能,包括容器启动事件、关闭事件等。实现了 ApplicationListener 事件监听接口的 Bean 可以接收到容器事件 , 并对事件进行响应处理 。 在 ApplicationContext 抽象实现类AbstractApplicationContext 中,我们可以发现存在一个 ApplicationEventMulticaster,它负责保存所有监听器,以便在容器产生上下文事件时通知这些事件监听者。
    • MessageSource:为应用提供 i18n 国际化消息访问的功能;
    • ResourcePatternResolver : 所 有 ApplicationContext 实现类都实现了类似于PathMatchingResourcePatternResolver 的功能,可以通过带前缀的 Ant 风格的资源文件路径装载 Spring 的配置文件。
    • LifeCycle:该接口是 Spring 2.0 加入的,该接口提供了 start()和 stop()两个方法,主要用于控制异步处理过程。在具体使用时,该接口同时被 ApplicationContext 实现及具体 Bean 实现, ApplicationContext 会将 start/stop 的信息传递给容器中所有实现了该接口的 Bean,以达到管理和控制 JMX、任务调度等目的。

    ApplicationContext接口的实现

    在考虑ApplicationContext接口的实现时,关键的点在于,不同Bean的配置方式(比如xml,groovy,annotation等)有着不同的资源加载方式,这便衍生除了众多ApplicationContext的实现类。

    第一,从类结构设计上看, 围绕着是否需要Refresh容器衍生出两个抽象类

    • GenericApplicationContext: 是初始化的时候就创建容器,往后的每次refresh都不会更改

    • AbstractRefreshableApplicationContext: AbstractRefreshableApplicationContext及子类的每次refresh都是先清除已有(如果不存在就创建)的容器,然后再重新创建;AbstractRefreshableApplicationContext及子类无法做到GenericApplicationContext混合搭配从不同源头获取bean的定义信息

    第二, 从加载的源来看(比如xml,groovy,annotation等), 衍生出众多类型的ApplicationContext, 典型比如:

    • FileSystemXmlApplicationContext: 从文件系统下的一个或多个xml配置文件中加载上下文定义,也就是说系统盘符中加载xml配置文件。
    • ClassPathXmlApplicationContext: 从类路径下的一个或多个xml配置文件中加载上下文定义,适用于xml配置的方式。
    • AnnotationConfigApplicationContext: 从一个或多个基于java的配置类中加载上下文定义,适用于java注解的方式。
    • ConfigurableApplicationContext: 扩展于 ApplicationContext,它新增加了两个主要的方法: refresh()和 close(),让 ApplicationContext 具有启动、刷新和关闭应用上下文的能力。在应用上下文关闭的情况下调用 refresh()即可启动应用上下文,在已经启动的状态下,调用 refresh()则清除缓存并重新装载配置信息,而调用close()则可关闭应用上下文。这些接口方法为容器的控制管理带来了便利,但作为开发者,我们并不需要过多关心这些方法。

    第三, 更进一步理解

    设计者在设计时AnnotationConfigApplicationContext为什么是继承GenericApplicationContext? 因为基于注解的配置,是不太会被运行时修改的,这意味着不需要进行动态Bean配置和刷新容器,所以只需要GenericApplicationContext。

    而基于XML这种配置文件,这种文件是容易修改的,需要动态性刷新Bean的支持,所以XML相关的配置必然继承AbstractRefreshableApplicationContext; 且存在多种xml的加载方式(位置不同的设计),所以必然会设计出AbstractXmlApplicationContext, 其中包含对XML配置解析成BeanDefination的过程。

    那么细心的你从上图可以发现AnnotationWebConfigApplicationContext却是继承了AbstractRefreshableApplicationContext而不是GenericApplicationContext, 为什么AnnotationWebConfigApplicationContext继承自AbstractRefreshableApplicationContext呢 ? 因为用户可以通过ApplicationContextInitializer来设置contextInitializerClasses(context-param / init-param), 在这种情况下用户倾向于刷新Bean的,所以设计者选择让AnnotationWebConfigApplicationContext继承了AbstractRefreshableApplicationContext。(如下是源码中Spring设计者对它的解释)

     * <p>As an alternative to setting the "contextConfigLocation" parameter, users may
     * implement an {@link org.springframework.context.ApplicationContextInitializer
     * ApplicationContextInitializer} and set the
     * {@linkplain ContextLoader#CONTEXT_INITIALIZER_CLASSES_PARAM "contextInitializerClasses"}
     * context-param / init-param. In such cases, users should favor the {@link #refresh()}
     * and {@link #scan(String...)} methods over the {@link #setConfigLocation(String)}
     * method, which is primarily for use by {@code ContextLoader}.
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    我们把之前的设计要点和设计结构结合起来看:

    到此,基本可以帮助你从顶层构建对IoC容器的设计理解,而不是过早沉溺于代码的细节; 所以《Java全栈知识体系》最大的目标是帮助你构筑体系化的认知,如果你自己去看源码而不站在顶层设计角度出发, 你多半会捡了芝麻丢了西瓜,时间一长啥印象没有。@pdai

    参考文章

    https://www.cnblogs.com/ITtangtang/p/3978349.html

    更多文章

    首先, 从Spring框架的整体架构和组成对整体框架有个认知。

    • Spring基础 - Spring和Spring框架组成
      • Spring是什么?它是怎么诞生的?有哪些主要的组件和核心功能呢? 本文通过这几个问题帮助你构筑Spring和Spring Framework的整体认知。

    其次,通过案例引出Spring的核心(IoC和AOP),同时对IoC和AOP进行案例使用分析。

    基于Spring框架和IOC,AOP的基础,为构建上层web应用,需要进一步学习SpringMVC。

    • Spring基础 - SpringMVC请求流程和案例
      • 前文我们介绍了Spring框架和Spring框架中最为重要的两个技术点(IOC和AOP),那我们如何更好的构建上层的应用呢(比如web 应用),这便是SpringMVC;Spring MVC是Spring在Spring Container Core和AOP等技术基础上,遵循上述Web MVC的规范推出的web开发框架,目的是为了简化Java栈的web开发。 本文主要介绍SpringMVC的请求流程和基础案例的编写和运行。

    Spring进阶 - IoC,AOP以及SpringMVC的源码分析

    • Spring进阶 - Spring IOC实现原理详解之IOC体系结构设计
      • 在对IoC有了初步的认知后,我们开始对IOC的实现原理进行深入理解。本文将帮助你站在设计者的角度去看IOC最顶层的结构设计
    • Spring进阶 - Spring IOC实现原理详解之IOC初始化流程
      • 上文,我们看了IOC设计要点和设计结构;紧接着这篇,我们可以看下源码的实现了:Spring如何实现将资源配置(以xml配置为例)通过加载,解析,生成BeanDefination并注册到IoC容器中的
    • Spring进阶 - Spring IOC实现原理详解之Bean实例化(生命周期,循环依赖等)
      • 上文,我们看了IOC设计要点和设计结构;以及Spring如何实现将资源配置(以xml配置为例)通过加载,解析,生成BeanDefination并注册到IoC容器中的;容器中存放的是Bean的定义即BeanDefinition放到beanDefinitionMap中,本质上是一个ConcurrentHashMap<String, Object>;并且BeanDefinition接口中包含了这个类的Class信息以及是否是单例等。那么如何从BeanDefinition中实例化Bean对象呢,这是本文主要研究的内容?
    • Spring进阶 - Spring AOP实现原理详解之切面实现
      • 前文,我们分析了Spring IOC的初始化过程和Bean的生命周期等,而Spring AOP也是基于IOC的Bean加载来实现的。本文主要介绍Spring AOP原理解析的切面实现过程(将切面类的所有切面方法根据使用的注解生成对应Advice,并将Advice连同切入点匹配器和切面类等信息一并封装到Advisor,为后续交给代理增强实现做准备的过程)。
    • Spring进阶 - Spring AOP实现原理详解之AOP代理
      • 上文我们介绍了Spring AOP原理解析的切面实现过程(将切面类的所有切面方法根据使用的注解生成对应Advice,并将Advice连同切入点匹配器和切面类等信息一并封装到Advisor)。本文在此基础上继续介绍,代理(cglib代理和JDK代理)的实现过程。
    • Spring进阶 - Spring AOP实现原理详解之Cglib代理实现
      • 我们在前文中已经介绍了SpringAOP的切面实现和创建动态代理的过程,那么动态代理是如何工作的呢?本文主要介绍Cglib动态代理的案例和SpringAOP实现的原理。
    • Spring进阶 - Spring AOP实现原理详解之JDK代理实现
      • 上文我们学习了SpringAOP Cglib动态代理的实现,本文主要是SpringAOP JDK动态代理的案例和实现部分。
    • Spring进阶 - SpringMVC实现原理之DispatcherServlet初始化的过程
      • 前文我们有了IOC的源码基础以及SpringMVC的基础,我们便可以进一步深入理解SpringMVC主要实现原理,包含DispatcherServlet的初始化过程和DispatcherServlet处理请求的过程的源码解析。本文是第一篇:DispatcherServlet的初始化过程的源码解析。
    • Spring进阶 - SpringMVC实现原理之DispatcherServlet处理请求的过程
      • 前文我们有了IOC的源码基础以及SpringMVC的基础,我们便可以进一步深入理解SpringMVC主要实现原理,包含DispatcherServlet的初始化过程和DispatcherServlet处理请求的过程的源码解析。本文是第二篇:DispatcherServlet处理请求的过程的源码解析。
  • 相关阅读:
    函数 lpad、rpad 不支持 length 函数运算改写
    FPGA学习笔记(十三)负数运算
    figlet - 字符画工具
    Asp .Net Core 系列:Asp .Net Core 集成 NLog
    [源码解析] NVIDIA HugeCTR,GPU 版本参数服务器---(7) ---Distributed Hash之前向传播
    GBASE 8s基本命令:onstat -k讲解
    EMQX Operator 如何快速创建弹性伸缩的 MQTT 集群
    java-php-python-ssm学生宿舍管理系统计算机毕业设计
    计算机毕业设计springboot教务管理系统4z3kl源码+系统+程序+lw文档+部署
    血氧仪方案组成结构设计分析
  • 原文地址:https://blog.csdn.net/sudadaipeng1/article/details/125477417