• Spring框架(二)Spring控制反转IoC详解


    一,什么是Spring IoC容器

    Spring是包含众多工具的IoC容器,控制反转即IoC,它有两个核心的功能:
    ①将对象(Bean)存储在容器(Spring)
    ②将对象(Bean)从容器(Spring)中取出来
    【扩展:将对象存放到容器中的好处】
    将对象存储在 IoC 容器相当于将以后可能⽤的所有工具制作好都放到仓库中,需要的时候直接取就行了,用完再把它放回到仓库。而new 对象的方式相当于,每次需要工具了,才现做,用完就扔掉了也不会保存,下次再用的时候还得重新做,这就是 IoC 容器和普通程序开发的区别

    二,IoC有哪些优点

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

    三,控制反转(IoC)有什么作用

    • 将创建对象的控制权交给Spring的IoC,以前需要程序员自己控制对象创建,现在交给Spring的IoC创建,如果需要使用需要通过DI(依赖注入)@Autowired自动注入
    • 解耦,由容器去维护具体的对象,降低耦合度

    【扩展:什么是解耦,代码示例】
    解耦指的是解决了代码的耦合性,耦合性也可以换⼀种叫法叫程序相关性。这就好比我们打造⼀辆完整的汽车,如果所有的配件都是自己造,那么当客户需求发生改变的时候,比如轮胎的尺寸不再是原来的尺寸了,那我们要自己动手来改了,但如果我们是把轮胎外包出去,那么即使是轮胎的尺寸发生变变了,我们只需要向代理工厂下订单就行了,我们自身身是不需要出力的

    • 在传统的代码中对象创建顺序是:Car -> Framework -> Bottom -> Tire
    • 改进之后解耦的代码的对象创建顺序是:Tire -> Bottom -> Framework -> Car在这里插入图片描述
    public class IocCarExample {
       public static void main(String[] args) {
         Tire tire = new Tire(20);
         Bottom bottom = new Bottom(tire);
         Framework framework = new Framework(bottom);
         Car car = new Car(framework);
         car.run();
      }
      //车类,把创建⼦类的⽅式,改为注⼊传递的⽅式
      static class Car {
        private Framework framework;
        public Car(Framework framework) {
          this.framework = framework;
        }
        public void run() {
          framework.init();
        }
      }
      //车身类
      static class Framework {
        private Bottom bottom;
        public Framework(Bottom bottom) {
          this.bottom = bottom;
        }
        public void init() {
        bottom.init();
        }
      }
    
    
    • 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

    四,IoC和DI有什么区别⭐

    IoC:控制反转,由主动new产生对象(耦合过高)转换成从外部提供对象,就是将对象的创建控制权从程序转移到了外部
    DI:依赖注入,就是在程序运行期间,自动的将一个对象从Spring拿出来给当前类使用

    • 区别:
      IoC 是“目标”也是⼀种思想,而目标和思想只是⼀种指导原则,而 DI 就是具体的实现
      例如:比如说我今天心情比较好,吃⼀顿好的犒劳犒劳自己,那么“吃⼀顿好的”是思想和目标(是 IoC),但最后我是吃海底捞,这就是具体的实现,就是 DI

    五,Spring IoC的实现机制⭐

    • 1, 简单工厂:通过一个方法传入一个标识,生产对应对象
    工厂模式案例:
    public static void main(String[] arge){
      // 调用工厂方法,根据传入参数,返回一个对象
      BaseService  userService = Factory.getBean("user");
    }
    class Factory{
      public static BaseService getBean(String beanName){
          if("user".equals(beanName)){
             return = new UserServiceImpl();
          }
          if("role".equals(beanName)){
             return = new RoleServiceImpl();
          }
       }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 2,反射:反射就是在工厂模式getBean()方法中通过反射的方式来创建Bean
    class Factory {
        public static Fruit getInstance(String ClassName) {
            Fruit f=null;
            try {
                f=(Fruit)Class.forName(ClassName).newInstance();
            } catch (Exception e) {
                e.printStackTrace();
            }
            return f;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    【扩展:反射是什么,反射的实现原理】
    反射机制是在运行的状态,对于任何一个类,都能知道所有属性和方法,对于任何一个对象都能调用任意方法属性,所以反射就是将Java类的各种成分映射成一个个对象

    • ①通过Class类的静态方法:forName(String className)(常用)获取Class对象
     try {
       Class stuClass3 = Class.forName("fanshe.Student");//注意此字符串必须是真实路径,就是带包名的类路径,包名.类名
       System.out.println(stuClass3 == stuClass2);//判断三种方式是否获取的是同一个Class对象
      } catch (ClassNotFoundException e) {
       e.printStackTrace();
      }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • ②通过反射获取构造方法并使用:
    public class Student {
      //---------------构造方法-------------------
     //(默认的构造方法)
     Student(String str){
      System.out.println("(默认)的构造方法 s = " + str);
     }
     //无参构造方法
     public Student(){
      System.out.println("调用了公有、无参构造方法执行了。。。");
     }
     
     //有一个参数的构造方法
     public Student(char name){
      System.out.println("姓名:" + name);
     }
     //有多个参数的构造方法
     public Student(String name ,int age){
      System.out.println("姓名:"+name+"年龄:"+ age);//这的执行效率有问题,以后解决。
     }
     //受保护的构造方法
     protected Student(boolean n){
      System.out.println("受保护的构造方法 n = " + n);
     }
     //私有构造方法
     private Student(int age){
      System.out.println("私有的构造方法   年龄:"+ age);
     }
     
    }
    
    • 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

    六,IoC支持哪些功能

    Spring 的 IoC 设计支持以下功能:

    • 依赖注入
    • 依赖检查
    • 自动装配
    • 支持集合
    • 指定初始化方法和销毁方法
    • 支持回调某些方法(但是需要实现 Spring 接口,略有侵入)
      其中,最重要的就是依赖注入,从 XML 的配置上说,即 ref 标签。对应 Spring RuntimeBeanReference 对象。
      对于 IoC 来说,最重要的就是容器。容器管理着 Bean 的生命周期,控制着 Bean 的依赖注入
      【扩展:解释Bean周期】下篇博客详细介绍

    七,BeanFactory和ApplicationContext有什么区别⭐

    首先创建Spring上下文的时候,会使用BeanFactory和ApplicationContext两种方法

    //先得到Spring上下文
      ApplicationContext context =
                    new ClassPathXmlApplicationContext("spring-config.xml");//配置文件对应
    
    • 1
    • 2
    • 3
     // 先得到 spring 获取 bean 的对象
      BeanFactory beanFactory =
                    new XmlBeanFactory(new ClassPathResource("spring-config.xml"));
    
    • 1
    • 2
    • 3
    • 共同点:BeanFactory和ApplicationContext是Spring的两大核心接口,都可以当做Spring的容器。其中ApplicationContext是BeanFactory的子接口
    • 区别:
      ①继承关系和功能:ApplicationContext是BeanFactory的子类,BeanFactory只是提供了基础操作Bean的方法,ApplicationContext除了继承了 BeanFactory 的所有功能之外,它还拥有独特的特性,还添加了对国际化⽀持,资源访问⽀持等
      ②性能:ApplicationContext 是⼀次性加载并初始化所有的 Bean 对象,而 BeanFactory是需要那个才去加载那个,因此更加轻量

    八,ApplicationContext通常的实现是什么

    • FileSystemXmlApplicationContext :此容器从一个XML文件中加载beans的定义,XML Bean 配置文件的全路径名必须提供给它的构造函数
    • ClassPathXmlApplicationContext:此容器也从一个XML文件中加载beans的定义,这里,你需要正确设置classpath因为这个容器将在classpath里找bean配置
    • WebXmlApplicationContext:此容器加载一个XML文件,此文件定义了一个WEB应用的所有bean

    九,依赖注入的方式,构造器依赖注入和 Setter方法注入的区别⭐

    • 依赖注入是时下最流行的IoC实现方式,依赖注入分为接口注入,Setter方法注入,和构造器注入,三种方式。其中接口注入由于在灵活性和易用性比较差,现在从Spring4开始已被废弃
    • 一、构造器注入
      将被依赖对象通过构造函数的参数注入给依赖对象,并且在初始化对象的时候注入。优点:对象初始化完成后便可获得可使用的对象。
      缺点:当需要注入的对象很多时,构造器参数列表将会很长;不够灵活。若有多种注入方式,每种方式只需注入指定几个依赖,那么就需要提供多个重载的构造函数,麻烦
    • 二、setter方法注入
      lOC Service Provider通过调用成员变量提供的setter函数将被依赖对象注入给依赖类优点:灵活,可以选择性地注入需要的对象
      缺点:依赖对象初始化完成后由于尚未注入被依赖对象,因此还不能使用
    • 三、接口注入
      依赖类必须要实现指定的接口,然后实现该接口中的一个函数,该函数就是用于依赖注入。该函数的参数就是要注入的对象
      优点:接口注入中,接口的名字、函数的名字都不重要,只要保证函数的参数是要注入的对象类型即可。缺点:侵入行太强,不建议使用
      【扩展:什么是侵入性?】
      如果类A要使用月别人提供的一个功能,若为了使用这功能,需要在自己的类中增加额外的代码,这就是侵入性
      区别:
      在这里插入图片描述

    十,依赖注入的基本原则和优点

    • 依赖注入的基本原则:
      应用组件不应该负责查找资源或者其他依赖的协作对象。配置对象的工作应该由IoC容器负责,“查找资源”的逻辑应该从应用组件的代码中抽取出来,交给IoC容器负责。容器全权负责组件的装配,它会把符合依赖关系的对象通过属性(JavaBean中的setter)或者是构造器传递给需要的对象
    • 优势:
      ①查找定位操作与应用代码完全无关
      ②不依赖于容器的API,可以很容易地在任何容器以外使用应用对象
      ③不需要特殊的接口,绝大多数对象可以做到完全不必依赖容器
  • 相关阅读:
    [附源码]JAVA毕业设计会议查询系统(系统+LW)
    笔记--es6
    电力物联网大数据平台架构及应用
    ceph 008 ceph多区域网关(ceph对象容灾) cephfs文件系统
    超强的苹果官网滚动文字特效实现
    论文辅助笔记:T2VEC一个疑虑:stackingGRUCell和GRU的区别在哪里?
    Java“牵手”天猫商品列表页数据采集+天猫商品价格数据排序,天猫API接口申请指南
    transformers的近期工作成果综述
    java中HashMap的实现原理
    【博客486】prometheus-----rate,irate,increase的原理
  • 原文地址:https://blog.csdn.net/qq_55660421/article/details/126211281