4.112 Spring的核心-控制反转IOC和依赖注入DI

    1. 什么是IOC控制反转,和依赖注入DI有什么区别

      1. 控制反转即IoC (Inversion of Control),它把传统上由程序代码直接操控的对象的调用权交给容器通过容器来实现对象组件的装配和管理。所谓的“控制反转”概念就是对组件对象控制权的转移,从程序代码本身转移到了外部容器。
      2. Spring IOC 负责创建对象,管理对象(通过依赖注入(DI),装配对象,配置对象,并且管理这些对象的整个生命周期。
    2. IOC具体作用

      1. 管理对象的创建和依赖关系的维护。对象的创建并不是一件简单的事,在对象关系比较复杂时,如果依赖关系需要程序猿来维护的话,那是相当头疼的
      2. 解耦,由容器去维护具体的对象
      3. 托管了类的产生过程,比如我们需要在类的产生过程中做一些处理,最直接的例子就是代理,如果有容器程序可以把这部分处理交给容器,应用程序则无需去关心类是如何完成代理的
      4. 依赖注入(DI):从 XML 的配置上说,即 ref 标签。对应 Spring RuntimeBeanReference 对象。
      5. 对于 IoC 来说,最重要的就是容器。容器管理着 Bean 的生命周期,控制着 Bean 的依赖注入
        1. 控制反转IoC是一个很大的概念,可以用不同的方式来实现。其主要实现方式有两种:依赖注入和依赖查找
        2. 依赖注入的基本原则是:应用组件不应该负责查找资源或者其他依赖的协作对象。配置对象的工作应该由IoC容器负责,“查找资源”的逻辑应该从应用组件的代码中抽取出来,交给IoC容器负责。容器全权负责组件的装配,它会把符合依赖关系的对象通过属性(JavaBean中的setter)或者是构造器传递给需要的对象。
        3. 依赖注入之所以更流行是因为它是一种更可取的方式:让容器全权负责依赖查询,受管组件只需要暴露JavaBean的setter方法或者带参数的构造器或者接口,使容器可以在初始化时组装对象的依赖关系。依赖注入DI与依赖查找方式相比主要优势
          1. 查找定位操作与应用代码完全无关。
          2. 不依赖于容器的API,可以很容易地在任何容器以外使用应用对象。
          3. 不需要特殊的接口,绝大多数对象可以做到完全不必依赖容器。
        4. 依赖注入实现方式
          1. 接口注入:接口注入由于在灵活性和易用性比较差,现在从Spring4开始已被废弃
          2. 构造器依赖注入:构造器依赖注入通过容器触发一个类的构造器来实现的,该类有一系列参数,每个参数代表一个对其他类的依赖。
          3. Setter方法注入:Setter方法注入是容器通过调用无参构造器或无参static工厂 方法实例化bean之后,调用该bean的setter方法,即实现了基于setter的依赖注入。
          4. 两种依赖方式都可以使用,构造器注入和Setter方法注入。最好的解决方案是用构造器参数实现强制依赖,setter方法实现可选依赖。
      6. 依赖检查
      7. 自动装配
      8. 支持集合
      9. 指定初始化方法和销毁方法
      10. 支持回调某些方法(但是需要实现 Spring 接口,略有侵入)
    3. IOC的优点

      1. IOC 或 依赖注入把应用的代码量降到最低
      2. 它使应用容易测试,单元测试不再需要单例和JNDI查找机制。
      3. 最小的代价和最小的侵入性使松散耦合得以实现。
      4. IOC容器支持加载服务时的饿汉式初始化和懒加载。
    4. IOC的实现原理

      1. Spring 中的 IoC 的实现原理就是工厂模式加反射机制。
    5. 实现容器的BeanFactory 和 ApplicationContext有什么区别?

      1. BeanFactory和ApplicationContext是Spring的两大核心接口,都可以当做Spring的容器。其中ApplicationContext是BeanFactory的子接口。
      2. BeanFactory:是Spring里面最底层的接口,包含了各种Bean的定义,读取bean配置文档,管理bean的加载、实例化,控制bean的生命周期,维护bean之间的依赖关系。
      3. ApplicationContext接口作为BeanFactory的派生,除了提供BeanFactory所具有的功能外,还提供了更完整的框架功能:
        1. 继承MessageSource,因此支持国际化。
        2. 统一的资源文件访问方式。
        3. 提供在监听器中注册bean的事件。
        4. 同时加载多个配置文件。
        5. 载入多个(有继承关系)上下文 ,使得每一个上下文都专注于一个特定的层次,比如应用的web层
      4. 加载方式
        1. BeanFactroy采用的是延迟加载形式来注入Bean的,即只有在使用到某个Bean时(调用getBean()),才对该Bean进行加载实例化。这样,我们就不能发现一些存在的Spring的配置问题。
        2. ApplicationContext,它是在容器启动时,一次性创建了所有的Bean。这样,在容器启动时,我们就可以发现Spring中存在的配置错误,这样有利于检查所依赖属性是否注入。
        3. 相对于基本的BeanFactory,ApplicationContext 唯一的不足是占用内存空间。当应用程序配置Bean较多时,程序启动较慢。
      5. 创建方式:BeanFactory通常以编程的方式被创建,ApplicationContext还能以声明的方式创建,如使用ContextLoader。
      6. 注册方式:BeanFactory和ApplicationContext都支持BeanPostProcessor、BeanFactoryPostProcessor的使用,但两者之间的区别是:BeanFactory需要手动注册,而ApplicationContext则是自动注册。
      7. 设计原理
        1. BeanFactory 简单粗暴,可以理解为就是个 HashMap,Key 是 BeanName,Value 是 Bean 实例。通常只提供注册(put),获取(get)这两个功能。我们可以称之为 “低级容器”。
        2. ApplicationContext 可以称之为 “高级容器”。因为他比 BeanFactory 多了更多的功能。他继承了多个接口。因此具备了更多的功能。例如资源的获取,支持多种消息(例如 JSP tag 的支持),对 BeanFactory 多了工具级别的支持等待。
原文地址:https://www.cnblogs.com/Smileing/p/13787154.html