Spring之容器:IOC(1)

简介: 【1月更文挑战第13天】一、IoC容器1、控制反转(IoC)2、依赖注入3、IoC容器在Spring的实现二、基于XML管理Bean1、搭建子模块spring6-ioc-xml2、实验一:获取bean①方式一:根据id获取②方式二:根据类型获取③方式三:根据id和类型④注意的地方⑤扩展知识3、实验二:依赖注入之setter注入4、实验三:依赖注入之构造器注入5、实验四:特殊值处理①字面量赋值②null值③xml实体④CDATA节6、实验五:为对象类型属性赋值方式一:引用外部bean方式二:内部bean方式三:级联属性赋值

文章目录

前言

一、IoC容器

1、控制反转(IoC)

2、依赖注入

3、IoC容器在Spring的实现

二、基于XML管理Bean

1、搭建子模块spring6-ioc-xml

2、实验一:获取bean

①方式一:根据id获取

②方式二:根据类型获取

③方式三:根据id和类型

④注意的地方

⑤扩展知识

3、实验二:依赖注入之setter注入

4、实验三:依赖注入之构造器注入

5、实验四:特殊值处理

①字面量赋值

②null值

③xml实体

④CDATA节

6、实验五:为对象类型属性赋值

方式一:引用外部bean

方式二:内部bean

方式三:级联属性赋值

总结


前言

IoC 是 Inversion of Control 的简写,译为“控制反转”,它不是一门技术,而是一种设计思想,是一个重要的面向对象编程法则,能够指导我们如何设计出松耦合、更优良的程序。

Spring 通过 IoC 容器来管理所有 Java 对象的实例化和初始化,控制对象与对象之间的依赖关系。我们将由 IoC 容器管理的 Java 对象称为 Spring Bean,它与使用关键字 new 创建的 Java 对象没有任何区别。

IoC 容器是 Spring 框架中最重要的核心组件之一,它贯穿了 Spring 从诞生到成长的整个过程。


一、IoC容器

1、控制反转(IoC)

  • 控制反转是一种思想。
  • 控制反转是为了降低程序耦合度,提高程序扩展力。
  • 控制反转,反转的是什么?
  • 将对象的创建权利交出去,交给第三方容器负责。
  • 将对象和对象之间关系的维护权交出去,交给第三方容器负责。
  • 控制反转这种思想如何实现呢?
  • DI(Dependency Injection):依赖注入

2、依赖注入

DI(Dependency Injection):依赖注入,依赖注入实现了控制反转的思想。

依赖注入:

  • 指Spring创建对象的过程中,将对象依赖属性通过配置进行注入

依赖注入常见的实现方式包括两种:

  • 第一种:set注入
  • 第二种:构造注入

所以结论是:IOC 就是一种控制反转的思想, 而 DI 是对IoC的一种具体实现。

Bean管理说的是:Bean对象的创建,以及Bean对象中属性的赋值(或者叫做Bean对象之间关系的维护)。

3、IoC容器在Spring的实现

Spring 的 IoC 容器就是 IoC思想的一个落地的产品实现。IoC容器中管理的组件也叫做 bean。在创建 bean 之前,首先需要创建IoC 容器。Spring 提供了IoC 容器的两种实现方式:

①BeanFactory

这是 IoC 容器的基本实现,是 Spring 内部使用的接口。面向 Spring 本身,不提供给开发人员使用。

②ApplicationContext

BeanFactory 的子接口,提供了更多高级特性。面向 Spring 的使用者,几乎所有场合都使用 ApplicationContext 而不是底层的 BeanFactory。

③ApplicationContext的主要实现类

类型名 简介
ClassPathXmlApplicationContext 通过读取类路径下的 XML 格式的配置文件创建 IOC 容器对象
FileSystemXmlApplicationContext 通过文件系统路径读取 XML 格式的配置文件创建 IOC 容器对象
ConfigurableApplicationContext ApplicationContext 的子接口,包含一些扩展方法 refresh() 和 close() ,让 ApplicationContext 具有启动、关闭和刷新上下文的能力。
WebApplicationContext 专门为 Web 应用准备,基于 Web 环境创建 IOC 容器对象,并将对象引入存入 ServletContext 域中。

二、基于XML管理Bean

1、搭建子模块spring6-ioc-xml

①搭建模块

搭建方式如:spring-first

②引入配置文件

引入spring-first模块配置文件:beans.xml、log4j2.xml

③添加依赖

<dependencies><!--spring context依赖--><!--当你引入Spring Context依赖之后,表示将Spring的基础依赖引入了--><dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>6.0.3</version></dependency><!--junit5测试--><dependency><groupId>org.junit.jupiter</groupId><artifactId>junit-jupiter-api</artifactId><version>5.3.1</version></dependency><!--log4j2的依赖--><dependency><groupId>org.apache.logging.log4j</groupId><artifactId>log4j-core</artifactId><version>2.19.0</version></dependency><dependency><groupId>org.apache.logging.log4j</groupId><artifactId>log4j-slf4j2-impl</artifactId><version>2.19.0</version></dependency></dependencies>

④引入java类

引入spring-first模块java及test目录下实体类

packagecom.atguigu.spring6.bean;
publicclassHelloWorld {
publicHelloWorld() {
System.out.println("无参数构造方法执行");
    }
publicvoidsayHello(){
System.out.println("helloworld");
    }
}
packagecom.atguigu.spring6.bean;
importorg.junit.jupiter.api.Test;
importorg.slf4j.Logger;
importorg.slf4j.LoggerFactory;
importorg.springframework.context.ApplicationContext;
importorg.springframework.context.support.ClassPathXmlApplicationContext;
publicclassHelloWorldTest {
privateLoggerlogger=LoggerFactory.getLogger(HelloWorldTest.class);
@TestpublicvoidtestHelloWorld(){
    }
}

2、实验一:获取bean

①方式一:根据id获取

由于 id 属性指定了 bean 的唯一标识,所以根据 bean 标签的 id 属性可以精确获取到一个组件对象。上个实验中我们使用的就是这种方式。

②方式二:根据类型获取

@TestpublicvoidtestHelloWorld1(){
ApplicationContextac=newClassPathXmlApplicationContext("beans.xml");
HelloWorldbean=ac.getBean(HelloWorld.class);
bean.sayHello();
}

③方式三:根据id和类型

@TestpublicvoidtestHelloWorld2(){
ApplicationContextac=newClassPathXmlApplicationContext("beans.xml");
HelloWorldbean=ac.getBean("helloorld", HelloWorld.class);
bean.sayHello();
}

④注意的地方

当根据类型获取bean时,要求IOC容器中指定类型的bean有且只能有一个

当IOC容器中一共配置了两个:

<beanid="helloworldOne"class="com.gedeshidai.spring6.bean.HelloWorld"></bean><beanid="helloworldTwo"class="com.gedeshidai.spring6.bean.HelloWorld"></bean>

根据类型获取时会抛出异常:

org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type ‘com.atguigu.spring6.bean.HelloWorld’ available: expected single matching bean but found 2: helloworldOne,helloworldTwo

⑤扩展知识

如果组件类实现了接口,根据接口类型可以获取 bean 吗?

可以,前提是bean唯一

如果一个接口有多个实现类,这些实现类都配置了 bean,根据接口类型可以获取 bean 吗?

不行,因为bean不唯一

结论

根据类型来获取bean时,在满足bean唯一性的前提下,其实只是看:『对象 instanceof 指定的类型』的返回结果,只要返回的是true就可以认定为和类型匹配,能够获取到。

java中,instanceof运算符用于判断前面的对象是否是后面的类,或其子类、实现类的实例。如果是返回true,否则返回false。也就是说:用instanceof关键字做判断时, instanceof 操作符的左右操作必须有继承或实现关系

3、实验二:依赖注入之setter注入

①创建学生类Student

packagecom.gedeshidai.spring6.bean;
publicclassStudent {
privateIntegerid;
privateStringname;
privateIntegerage;
privateStringsex;
publicStudent() {
    }
publicIntegergetId() {
returnid;
    }
publicvoidsetId(Integerid) {
this.id=id;
    }
publicStringgetName() {
returnname;
    }
publicvoidsetName(Stringname) {
this.name=name;
    }
publicIntegergetAge() {
returnage;
    }
publicvoidsetAge(Integerage) {
this.age=age;
    }
publicStringgetSex() {
returnsex;
    }
publicvoidsetSex(Stringsex) {
this.sex=sex;
    }
@OverridepublicStringtoString() {
return"Student{"+"id="+id+", name='"+name+'\''+", age="+age+", sex='"+sex+'\''+'}';
    }
}

②配置bean时为属性赋值

spring-di.xml

<beanid="studentOne"class="com.gedeshidai.spring6.bean.Student"><!-- property标签:通过组件类的setXxx()方法给组件对象设置属性 --><!-- name属性:指定属性名(这个属性名是getXxx()、setXxx()方法定义的,和成员变量无关) --><!-- value属性:指定属性值 --><propertyname="id"value="1001"></property><propertyname="name"value="张三"></property><propertyname="age"value="23"></property><propertyname="sex"value="男"></property></bean>

③测试

@TestpublicvoidtestDIBySet(){
ApplicationContextac=newClassPathXmlApplicationContext("spring-di.xml");
StudentstudentOne=ac.getBean("studentOne", Student.class);
System.out.println(studentOne);
}

4、实验三:依赖注入之构造器注入

①在Student类中添加有参构造

publicStudent(Integerid, Stringname, Integerage, Stringsex) {
this.id=id;
this.name=name;
this.age=age;
this.sex=sex;
}

②配置bean

spring-di.xml

<beanid="studentTwo"class="com.gedeshidai.spring6.bean.Student"><constructor-argvalue="1002"></constructor-arg><constructor-argvalue="李四"></constructor-arg><constructor-argvalue="33"></constructor-arg><constructor-argvalue="女"></constructor-arg></bean>

注意:

constructor-arg标签还有两个属性可以进一步描述构造器参数:

  • index属性:指定参数所在位置的索引(从0开始)
  • name属性:指定参数名

③测试

@TestpublicvoidtestDIByConstructor(){
ApplicationContextac=newClassPathXmlApplicationContext("spring-di.xml");
StudentstudentOne=ac.getBean("studentTwo", Student.class);
System.out.println(studentOne);
}

5、实验四:特殊值处理

①字面量赋值

什么是字面量?

int a = 10;

声明一个变量a,初始化为10,此时a就不代表字母a了,而是作为一个变量的名字。当我们引用a的时候,我们实际上拿到的值是10。

而如果a是带引号的:‘a’,那么它现在不是一个变量,它就是代表a这个字母本身,这就是字面量。所以字面量没有引申含义,就是我们看到的这个数据本身。

<!-- 使用value属性给bean的属性赋值时,Spring会把value属性的值看做字面量 --><propertyname="name"value="张三"/>

②null值

<propertyname="name"><null/></property>

注意:

<propertyname="name"value="null"></property>

以上写法,为name所赋的值是字符串null

③xml实体

<!-- 小于号在XML文档中用来定义标签的开始,不能随便使用 --><!-- 解决方案一:使用XML实体来代替 --><propertyname="expression"value="a &lt; b"/>

④CDATA节

<propertyname="expression"><!-- 解决方案二:使用CDATA节 --><!-- CDATA中的C代表Character,是文本、字符的含义,CDATA就表示纯文本数据 --><!-- XML解析器看到CDATA节就知道这里是纯文本,就不会当作XML标签或属性来解析 --><!-- 所以CDATA节中写什么符号都随意 --><value><![CDATA[a < b]]></value></property>

6、实验五:为对象类型属性赋值

①创建班级类Clazz

packagecom.gedeshidai.spring6.beanpublicclassClazz {
privateIntegerclazzId;
privateStringclazzName;
publicIntegergetClazzId() {
returnclazzId;
    }
publicvoidsetClazzId(IntegerclazzId) {
this.clazzId=clazzId;
    }
publicStringgetClazzName() {
returnclazzName;
    }
publicvoidsetClazzName(StringclazzName) {
this.clazzName=clazzName;
    }
@OverridepublicStringtoString() {
return"Clazz{"+"clazzId="+clazzId+", clazzName='"+clazzName+'\''+'}';
    }
publicClazz() {
    }
publicClazz(IntegerclazzId, StringclazzName) {
this.clazzId=clazzId;
this.clazzName=clazzName;
    }
}

②修改Student类

在Student类中添加以下代码:

privateClazzclazz;
publicClazzgetClazz() {
returnclazz;
}
publicvoidsetClazz(Clazzclazz) {
this.clazz=clazz;
}

方式一:引用外部bean

配置Clazz类型的bean:

<beanid="clazzOne"class="com.atguigu.spring6.bean.Clazz"><propertyname="clazzId"value="1111"></property><propertyname="clazzName"value="财源滚滚班"></property></bean>

为Student中的clazz属性赋值:

<beanid="studentFour"class="com.gedeshidai.spring6.bean.Student"><propertyname="id"value="1004"></property><propertyname="name"value="赵六"></property><propertyname="age"value="26"></property><propertyname="sex"value="女"></property><!-- ref属性:引用IOC容器中某个bean的id,将所对应的bean为属性赋值 --><propertyname="clazz"ref="clazzOne"></property></bean>

错误演示:

<beanid="studentFour"class="com.gedeshidai.spring6.bean.Student"><propertyname="id"value="1004"></property><propertyname="name"value="赵六"></property><propertyname="age"value="26"></property><propertyname="sex"value="女"></property><propertyname="clazz"value="clazzOne"></property></bean>

如果错把ref属性写成了value属性,会抛出异常: Caused by: java.lang.IllegalStateException: Cannot convert value of type ‘java.lang.String’ to required type ‘com.atguigu.spring6.bean.Clazz’ for property ‘clazz’: no matching editors or conversion strategy found

意思是不能把String类型转换成我们要的Clazz类型,说明我们使用value属性时,Spring只把这个属性看做一个普通的字符串,不会认为这是一个bean的id,更不会根据它去找到bean来赋值

方式二:内部bean

<beanid="studentFour"class="com.gedeshidai.spring6.bean.Student"><propertyname="id"value="1004"></property><propertyname="name"value="赵六"></property><propertyname="age"value="26"></property><propertyname="sex"value="女"></property><propertyname="clazz"><!-- 在一个bean中再声明一个bean就是内部bean --><!-- 内部bean只能用于给属性赋值,不能在外部通过IOC容器获取,因此可以省略id属性 --><beanid="clazzInner"class="com.gedeshidai.spring6.bean.Clazz"><propertyname="clazzId"value="2222"></property><propertyname="clazzName"value="远大前程班"></property></bean></property></bean>

方式三:级联属性赋值

<beanid="studentFour"class="com.gedeshidai.spring6.bean.Student"><propertyname="id"value="1004"></property><propertyname="name"value="赵六"></property><propertyname="age"value="26"></property><propertyname="sex"value="女"></property><propertyname="clazz"ref="clazzOne"></property><propertyname="clazz.clazzId"value="3333"></property><propertyname="clazz.clazzName"value="最强王者班"></property></bean>

总结

以上就是Spring之容器:IOC(1)的相关知识点,希望对你有所帮助。

积跬步以至千里,积怠惰以至深渊。时代在这跟着你一起努力哦!

相关文章
|
3月前
|
XML Java 测试技术
《深入理解Spring》:IoC容器核心原理与实战
Spring IoC通过控制反转与依赖注入实现对象间的解耦,由容器统一管理Bean的生命周期与依赖关系。支持XML、注解和Java配置三种方式,结合作用域、条件化配置与循环依赖处理等机制,提升应用的可维护性与可测试性,是现代Java开发的核心基石。
|
3月前
|
XML Java 应用服务中间件
【SpringBoot(一)】Spring的认知、容器功能讲解与自动装配原理的入门,带你熟悉Springboot中基本的注解使用
SpringBoot专栏开篇第一章,讲述认识SpringBoot、Bean容器功能的讲解、自动装配原理的入门,还有其他常用的Springboot注解!如果想要了解SpringBoot,那么就进来看看吧!
485 2
|
11月前
|
XML Java 测试技术
Spring IOC—基于注解配置和管理Bean 万字详解(通俗易懂)
Spring 第三节 IOC——基于注解配置和管理Bean 万字详解!
810 26
|
7月前
|
XML 人工智能 Java
Spring IOC 到底是什么?
IOC(控制反转)是一种设计思想,主要用于解耦代码,简化依赖管理。其核心是将对象的创建和管理交给容器处理,而非由程序直接硬编码实现。通过IOC,开发者无需手动new对象,而是由框架负责实例化、装配和管理依赖对象。常见应用如Spring框架中的BeanFactory和ApplicationContext,它们实现了依赖注入和动态管理功能,提升了代码的灵活性与可维护性。
206 1
|
8月前
|
XML Java 数据格式
Spring IoC容器的设计与实现
Spring 是一个功能强大且模块化的 Java 开发框架,其核心架构围绕 IoC 容器、AOP、数据访问与集成、Web 层支持等展开。其中,`BeanFactory` 和 `ApplicationContext` 是 Spring 容器的核心组件,分别定位为基础容器和高级容器,前者提供轻量级的 Bean 管理,后者扩展了事件发布、国际化等功能。
|
XML Java 数据格式
京东一面:spring ioc容器本质是什么? ioc容器启动的步骤有哪些?
京东一面:spring ioc容器本质是什么? ioc容器启动的步骤有哪些?
|
10月前
|
Java 容器 Spring
什么是Spring IOC 和DI ?
IOC : 控制翻转 , 它把传统上由程序代码直接操控的对象的调用权交给容 器,通过容器来实现对象组件的装配和管理。所谓的“控制反转”概念就是对组件对象控制权的转 移,从程序代码本身转移到了外部容器。 DI : 依赖注入,在我们创建对象的过程中,把对象依赖的属性注入到我们的类中。
|
11月前
|
XML Java 数据格式
Spring容器的本质
本文主要讨论Spring容器最核心的机制,用最少的代码讲清楚Spring容器的本质。
|
XML Java 数据格式
Spring5入门到实战------7、IOC容器-Bean管理XML方式(外部属性文件)
这篇文章是Spring5框架的实战教程,主要介绍了如何在Spring的IOC容器中通过XML配置方式使用外部属性文件来管理Bean,特别是数据库连接池的配置。文章详细讲解了创建属性文件、引入属性文件到Spring配置、以及如何使用属性占位符来引用属性文件中的值。
Spring5入门到实战------7、IOC容器-Bean管理XML方式(外部属性文件)
|
XML Java 数据格式
Spring5系列学习文章分享---第一篇(概述+特点+IOC原理+IOC并操作之bean的XML管理操作)
Spring5系列学习文章分享---第一篇(概述+特点+IOC原理+IOC并操作之bean的XML管理操作)
143 1