当前位置: 首页 > news >正文

【SSM】spring核心思想——IOC和DI

文章目录

  • 1. Spring 是什么?
  • 2. IOC 是什么
    • 2.1 正向控制
    • 2.2 反向控制
  • 3. DI 概念说明
  • 4. Spring 的工作模式可以理解成一个买卖对象的市场
  • 5. 以一个小程序来理解 IOC 和 DI
  • 6. 以 XML 文件的方式将对象注入到我们自己创建的库中
    • 6.1 新建一个 XML 文件
    • 6.2 在 Main 中操作这个库
    • 6.2 XML 文件的方式注入 bean(对象)

1. Spring 是什么?

我们通常所说的 Spring 指的是 Spring Framework(Spring 框架),它是一个开源框架,有着活跃而庞 大的社区,这就是它之所以能长久不衰的原因。Spring 支持广泛的应用场景,它可以让 Java 企业级的应用程序开发起来更简单。

一句话概括 Spring :Spring 是包含了众多工具方法的 IOC 容器

那么什么是 IOC 呢

2. IOC 是什么

IOC (Inversion of Control 控制反转),就是说 Spring 是一个控制反转的容器

既然提到了控制反转,那么就来看看什么是正向控制,什么是反向控制

2.1 正向控制

在我们之前的大部分代码中,当我们要用到一个对象时,我们就需要自己构造这个对象,自己 new 对象,自己造,自己用,典型的自给自足生产模式

2.2 反向控制

在 Spring 的大部分代码中,我们只需要声明我们需要一个什么样的对象,对象的构造我们不管,这里的对象就是 Spring 帮我们注入进来的,对象就来自于 IOC 容器

3. DI 概念说明

说到 IoC 不得不提的一个词就是“DI”,DI 是 Dependency Injection 的缩写,翻译成中文是“依赖注入”的 意思。

所谓依赖注入,就是由 IoC 容器在运行期间,动态地将某种依赖关系注入到对象之中。所以,依赖注入 (DI)和控制反转(IoC)是从不同的角度的描述的同一件事情,就是指通过引入 IoC 容器,利用依赖关系注入的方式,实现对象之间的解耦。

4. Spring 的工作模式可以理解成一个买卖对象的市场

Spring 容器就是这个市场,我们既可以是卖家,也可以是买家

当我们是卖家时,将我们的对象注入到 Spring 库

当我们是买家时,声明我们需要的对象,Spring 从库中将对象取出来给我们

请添加图片描述

5. 以一个小程序来理解 IOC 和 DI

当我们需要造一辆车时,就需要得到一个四个部分,才能造成一辆车

以代码实现时,我们就需要这四个对象,先手动将这四个对象注入到库中

请添加图片描述

@Component:将被修饰的类注入到库中

@Autowired:自动注入,在这里可以理解成从库中拿到所修饰的元素

这里重写 toSpring() 只是为了打印对象,没有其他特殊意义

使用 context.getBeanDefinitionNames() 得到库中所有对象并打印后,可以清楚的看到,这四个对象已经被注入到库中

后续 car 对象也成功打印

@Component
public class Car {
    @Autowired
    private Framework framework;

    @Override
    public String toString() {
        return "Car{" +
                "framework=" + framework +
                '}';
    }
}
@Component
public class Framework {
    @Autowired
    private Bottom bottom;

    @Override
    public String toString() {
        return "Framework{" +
                "bottom=" + bottom +
                '}';
    }
}
@Component
public class Bottom {
    @Autowired
    private Tire t1;
    @Autowired
    private Tire t2;
    @Autowired
    private Tire t3;
    @Autowired
    private Tire t4;

    @Override
    public String toString() {
        return "Bottom{" +
                "t1=" + t1 +
                ", t2=" + t2 +
                ", t3=" + t3 +
                ", t4=" + t4 +
                '}';
    }
}
@Component
public class Tire {
    @Override
    public String toString() {
        return "轮子";
    }
}

主类

@SpringBootApplication
public class IocApplication {

   public static void main(String[] args) {
      ConfigurableApplicationContext context = SpringApplication.run(IocApplication.class, args);

      Car car = context.getBean(Car.class);

      // 得到库中所有的 bean(对象)
      String[] beanDefinitionNames = context.getBeanDefinitionNames();
      for (String beanDefinitionName : beanDefinitionNames) {
         System.out.println(beanDefinitionName);
      }
       
      System.out.println(car);
   }

}

请添加图片描述

请添加图片描述

当有一个对象没有被 @Component 修饰,即库中没有这个对象,那么其他对象就不能用@Autowired 得到这个对象,则就会出现以下异常

请添加图片描述

6. 以 XML 文件的方式将对象注入到我们自己创建的库中

6.1 新建一个 XML 文件

现在这个 XML 文件就可以看成是一个空的 Spring 库

请添加图片描述

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd">
</beans>

6.2 在 Main 中操作这个库

用 FileSystemXmlApplicationContext() 打开这个 XML 文件之后,就可以在这里对刚刚创建的 Spring 进行一些操作

public class Main {
    public static void main(String[] args) {
        //1. 以 XML 配置文件手动创建 Spring 容器
        FileSystemXmlApplicationContext context = new FileSystemXmlApplicationContext("spring-config.xml");

        // 2. 打印容器中所有对象的 bean-id
        String[] names = context.getBeanDefinitionNames();
        System.out.println("所有对象陈列:");
        for (String name : names) {
            Object bean = context.getBean(name);

            // 得到此类的全名,即包含包名的名称
            String canonicalName = bean.getClass().getCanonicalName();

            System.out.println(name + "      " + canonicalName);
        }
        int count = context.getBeanDefinitionCount();
        System.out.println("一共有" + count + "个");
    }
}

6.2 XML 文件的方式注入 bean(对象)

以 bean 标签来对对象进行注入,以下 XML 用不同的三种方法注入了三个对象

打印后可以清楚的看到,三个对象成功注入

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd">

<!--    xml的方式注入类,new 的过程也交给 Spring-->
    <bean id="empty-class" class="com.hsq.ioc2.EmptyClass"/>

<!--    构造方法注入类-->
    <bean id="second-class" class="com.hsq.ioc2.SecondClass">
<!--        依赖 EmptyClass 注入 SecondClass-->
        <constructor-arg ref="empty-class"/>
    </bean>

<!--    set 方法注入类-->
    <bean id="third-class" class="com.hsq.ioc2.ThirdClass">
        <property name="EmptyClass" ref="empty-class"/>
        <property name="SecondClass" ref="second-class"/>
    </bean>
</beans>
public class EmptyClass {
    public EmptyClass() {
        System.out.println("被调用");
    }
}
public class SecondClass {
    public SecondClass(EmptyClass emptyClass) {
        System.out.println("second 被调用");
    }
}
public class ThirdClass {
    public EmptyClass emptyClass;
    public SecondClass secondClass;

    public void setEmptyClass(EmptyClass emptyClass) {
        System.out.println("setEmptyClass() " + emptyClass);
        this.emptyClass = emptyClass;
    }

    public void setSecondClass(SecondClass secondClass) {
        System.out.println("setSecondClass() " + secondClass);
        this.secondClass = secondClass;
    }

    public ThirdClass() {
        System.out.println("third 被调用");
    }
}

请添加图片描述

相关文章:

  • 自由的程序员应该学会自由地控制空间-----动态内存管理
  • [架构之路-51]:架构师 - 用系统化、结构化思维解决复杂难搞的软件故障问题 - 马克思主义哲学在软件系统中的应用
  • 【面试题】JavaScript数组切片方法有哪些?
  • 【PyTorch深度学习项目实战100例】—— 基于CNN卷积神经网络实现中文手写汉字识别 | 第60例
  • HarmonyOS系统中内核实现UART串口通信方法
  • Selenium4.0+Python三种元素等待方式介绍 及 元素等待封装
  • django梳理
  • 嵌入式软件调试的发展历程
  • PT_连续型随机变量/分布函数/概率密度
  • Python告别pip手动安装模块,实现全自动安装第三方库,彻底解放你的双手
  • 文件目录操作——Linux命令核心
  • Taichi 加速 Python 中图像处理
  • Vue--》MVVM模型在Vue中的使用
  • 迷宫求解(云南大学)
  • 【夜读】坚持这5个习惯,遇见更优秀的自己
  • 【comparator, comparable】小总结
  • EOS是什么
  • GDB 调试 Mysql 实战(三)优先队列排序算法中的行记录长度统计是怎么来的(上)...
  • JAVA SE 6 GC调优笔记
  • linux学习笔记
  • maya建模与骨骼动画快速实现人工鱼
  • node和express搭建代理服务器(源码)
  • React 快速上手 - 06 容器组件、展示组件、操作组件
  • 百度地图API标注+时间轴组件
  • 程序员该如何有效的找工作?
  • 从零搭建Koa2 Server
  • 关键词挖掘技术哪家强(一)基于node.js技术开发一个关键字查询工具
  • 基于Dubbo+ZooKeeper的分布式服务的实现
  • 人脸识别最新开发经验demo
  • 什么软件可以提取视频中的音频制作成手机铃声
  • FaaS 的简单实践
  • 分布式关系型数据库服务 DRDS 支持显示的 Prepare 及逻辑库锁功能等多项能力 ...
  • ​ 全球云科技基础设施:亚马逊云科技的海外服务器网络如何演进
  • ​软考-高级-信息系统项目管理师教程 第四版【第14章-项目沟通管理-思维导图】​
  • # MySQL server 层和存储引擎层是怎么交互数据的?
  • # Swust 12th acm 邀请赛# [ K ] 三角形判定 [题解]
  • ### Error querying database. Cause: com.mysql.jdbc.exceptions.jdbc4.CommunicationsException
  • ###项目技术发展史
  • (2)关于RabbitMq 的 Topic Exchange 主题交换机
  • (ctrl.obj) : error LNK2038: 检测到“RuntimeLibrary”的不匹配项: 值“MDd_DynamicDebug”不匹配值“
  • (Mirage系列之二)VMware Horizon Mirage的经典用户用例及真实案例分析
  • (八)Flask之app.route装饰器函数的参数
  • (附源码)ssm高校实验室 毕业设计 800008
  • (介绍与使用)物联网NodeMCUESP8266(ESP-12F)连接新版onenet mqtt协议实现上传数据(温湿度)和下发指令(控制LED灯)
  • (三)mysql_MYSQL(三)
  • (四)搭建容器云管理平台笔记—安装ETCD(不使用证书)
  • (转)LINQ之路
  • .bat批处理出现中文乱码的情况
  • .NET C# 使用 SetWindowsHookEx 监听鼠标或键盘消息以及此方法的坑
  • .net 开发怎么实现前后端分离_前后端分离:分离式开发和一体式发布
  • /deep/和 >>>以及 ::v-deep 三者的区别
  • :如何用SQL脚本保存存储过程返回的结果集
  • [20171102]视图v$session中process字段含义
  • [2019.2.28]BZOJ4033 [HAOI2015]树上染色
  • [Android Studio 权威教程]断点调试和高级调试