【Spring】核心部分之IOC:通过列举代码例子,从底层刨析,深入源码,轻轻松松理解Spring的核心IOC,IOC有这一篇足以

简介: 【Spring】核心部分之IOC:通过列举代码例子,从底层刨析,深入源码,轻轻松松理解Spring的核心IOC,IOC有这一篇足以

@TOC

一,概念和原理

1.概念:控制反转

IOC:控制反转,把对象创建和对象之间的调用过程,交给 Spring 进行管理
使用 IOC 目的:为了耦合度降低,简化开发

2.原理:(xml解析,工厂模式,反射)

xml 解析:
工厂模式:
反射:关于反射内容
Java中反射的理解及如何使用——精简

3.IOC接口:两个主要容器实现类(加载xml配置文件)

BeanFactory:IOC 容器基本实现,是 Spring 内部的使用接口,不提供开发人员进行使用,加载配置文件时候不会创建对象,在获取对象(使用)才去创建对象
ApplicationContext:BeanFactory 接口的子接口,提供更多更强大的功能,一般由开发人员进行使用加载配置文件时候就会把在配置文件对象进行创建

当写好xml配置文件后,通过加载配置文件来创建对象

在这里插入图片描述

代码举例

public class TestSpring {
      
    
    
    /**
     * bean1.xml:src目录下的配置文件
     * User:类
     * user:配置文件中的id,也就是User类的别名
     */
    @Test
    public void testUser(){
      
    
    
       //1.加载spring的配置文件
        ApplicationContext context=new ClassPathXmlApplicationContext("bean1.xml") ;
        //2.获取配置创建的对象
        User user = context.getBean("user", User.class);
    }
}

二,操作Bean管理

Bean 管理指的是两个操作
(1)Spring 创建对象
(2)Spirng 注入属性

1.基于 xml 配置文件方式实现

(1)基于 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 http://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="studentOne" class="com.li.Student"></bean>
</beans>

在 spring 配置文件中,使用 bean 标签,标签里面添加对应属性,就可以实
现对象创建.在 bean 标签有很多属性
id 属性:唯一标识
class 属性:类全路径(包类路径)

创建对象时候,默认也是执行无参数构造方法完成对象创建

代码举例:

<bean id="studentOne" class="com.li.Student"></bean>

获取对象的三种方式:

1.根据bean的id来获取

Student studentOne = (Student) context1.getBean("studentOne");

2.根据bean的类型

Student studentOne = context1.getBean(Student.class);

当根据bean的类型获取时,容器中有且只能有一个类型匹配的bean

3.根据bean的id和类型来获取

Student studentOne = context1.getBean("studentOne", Student.class);

(2)基于 xml 方式注入属性

DI:依赖注入,就是注入属性

方式一:使用 set 方法进行注入

1.需要提供属性对应的set方法

public class Book {
        
      
      
    private String bname;
    private String bauthor;

    public void setBname(String bname) {
        
      
      
        this.bname = bname;
    }

    public void setBauthor(String bauthor) {
        
      
      
        this.bauthor = bauthor;
    }

    public Book() {
        
      
      
    }
}

2.在 spring 配置文件配置对象创建,配置属性注入
使用 property 完成属性注入
name:类里面属性名称
value:向属性注入的值

<bean id="book" class="com.li.Spring.Book">
        <property name="bname" value="及你太美"></property>
        <property name="bauthor" value="蔡徐坤"></property>
    </bean>

方式二:使用有参数构造进行注入

1.创建类,定义属性,创建属性对应有参数构造方法

public class Book {
        
      
      
    private String bname;
    private String bauthor;


    public Book(String bname, String bauthor) {
        
      
      
        this.bname = bname;
        this.bauthor = bauthor;
    }
    }

2.在 spring 配置文件中进行配置

<bean id="book" class="com.li.Spring.Book">
        <constructor-arg name="bname" value="及你太美"></constructor-arg>
        <constructor-arg name="bauthor" value="蔡徐坤"></constructor-arg>
    </bean>

补充:xml 注入其他类型属性

1.字面量

null:

<bean id="book" class="com.li.Spring.Book">
        <property name="bname" value="葵花宝典"></property>
        <property name="bauthor" >
            <null/>
        </property>
    </bean>

属性值包含特殊符号:

<property name="bname">
 <value><![CDATA[<<蔡徐坤>>]]></value>
</property>

2.注入属性-外部 bean

(1)创建两个类 service 类和 dao 类
(2)在 service 调用 dao 里面的方法
(3)在 spring 配置文件中进行配置

<bean id="service" class="com.li.Spring.Service">
    <property name="dao" ref="dao"></property>
</bean>
    <bean id="dao" class="com.li.Spring.Dao"></bean>

3.注入属性-内部 bean

<bean id="emp" class="com.li.spring.bean.Emp">
 <!--设置两个普通属性-->
 <property name="ename" value="lucy"></property>
 <property name="gender" value="女"></property>
 <!--设置对象类型属性-->
 <property name="dept">
 <bean id="dept" class="com.li.spring.bean.Dept">
 <property name="dname" value="安保部"></property>
 </bean>
 </property>
</bean>

4.注入属性-级联赋值

第一种写法:

<bean id="emp" class="com.li.spring.bean.Emp">
 <!--设置两个普通属性-->
 <property name="ename" value="lucy"></property>
 <property name="gender" value="女"></property>
 <!--级联赋值-->
 <property name="dept" ref="dept"></property>
</bean>
<bean id="dept" class="com.li.spring.bean.Dept">
 <property name="dname" value="财务部"></property>
</bean>

第二种写法

<bean id="emp" class="com.li.spring.bean.Emp">
 <!--设置两个普通属性-->
 <property name="ename" value="lucy"></property>
 <property name="gender" value="女"></property>
 <!--级联赋值-->
 <property name="dept" ref="dept"></property>
 <property name="dept.dname" value="技术部"></property>
</bean>
<bean id="dept" class="com.li.spring.bean.Dept">
 <property name="dname" value="财务部"></property>
</bean>

5.xml 注入集合属性

1、注入数组类型属性
2、注入 List 集合类型属性
3、注入 Map 集合类型属性

<bean id="stu" class="com.li.spring.collectiontype.Stu">
 <!--数组类型属性注入-->
 <property name="courses">
 <array>
 <value>java 课程</value>
 <value>数据库课程</value>
 </array>
 </property>
 <!--list 类型属性注入-->
 <property name="list">
 <list>
 <value>张三</value>
 <value>小三</value>
 </list>
 </property>
 <!--map 类型属性注入-->
 <property name="maps">
 <map>
 <entry key="JAVA" value="java"></entry>
 <entry key="PHP" value="php"></entry>
 </map>
 </property>
 <!--set 类型属性注入-->
 <property name="sets">
 <set>
 <value>MySQL</value>
 <value>Redis</value>
 </set>
 </property>
</bean>

<对象属性值>

<bean id="course1" class="com.li.spring.collectiontype.Course">
 <property name="cname" value="Spring 框架"></property>
</bean>
<bean id="course2" class="com.li.spring.collectiontype.Course">
 <property name="cname" value="MyBatis 框架"></property>
</bean>
<!--注入 list 集合类型,值是对象-->
<property name="courseList">
 <list>
 <ref bean="course1"></ref>
 <ref bean="course2"></ref>
 </list>
</property>

方式三:p 名称空间注入

使用 p 名称空间注入,可以简化基于 xml 配置方式
1.添加 p 名称空间在配置文件中
2.进行属性注入,在 bean 标签里面进行操作

方式四:xml 自动装配

自动装配:根据当前指定的策略 ,在ioc容器中匹配某个bean,自动为bean中的类的属性或接口类型的属性赋值
可以通过bean标签中的autowire属性自动装配
no,default:表示不装配

传统的注入方法:

<bean id="userController" class="com.li.controller.userController">
       <property name="userservice" ref="userviceimp"></property>
   </bean>

    <bean id="userviceimp" class="com.li.service.imp.userviceimp">
        <property name="userdao" ref="userdao"></property>
    </bean>

    <bean id="userdao" class="com.li.dao.imp.userdaoimp"></bean>

byType:
根据要赋值属性的类型,在ioc容器中匹配某个bean,为属性赋值

 <bean id="userController" class="com.li.controller.userController" autowire="byType">
   </bean>
    <bean id="userviceimp" class="com.li.service.imp.userviceimp" autowire="byType">
    </bean>
    <bean id="userdao" class="com.li.dao.imp.userdaoimp"></bean>

byName
一般不用
将要赋值的属性的属性名作为bean的id在容器中匹配某个bean,为属性赋值

<bean id="userController" class="com.li.controller.userController" autowire="byType">
   </bean>
    <bean id="userviceimp" class="com.li.service.imp.userviceimp" autowire="byName">
    </bean>
    <bean id="userdao" class="com.li.dao.imp.userdaoimp"></bean>
    <bean id="dao" class="com.li.dao.imp.userdaoimp"></bean>

2.基于注解方式实现

什么是注解
(1)注解是代码特殊标记,格式:@注解名称(属性名称=属性值, 属性名称=属性值..)
(2)使用注解,注解作用在类上面,方法上面,属性上面
(3)使用注解目的:简化 xml 配置

(1)基于注解方式实现对象创建

Spring 针对 Bean 管理中创建对象提供注解
(1)@Component:将类标识为普通组件
(2)@Service:将类标识为业务层组件
(3)@Controller:将类标识为控制层组件
(4)@Repository:将类标识为持久层组件
上面四个注解功能是一样的,都可以用来创建 bean 实例

举例:

第一步:开启组件扫描

<context:component-scan base-package="com.li.spring"></context:component-scan>

第二步:创建类,在类上面添加创建对象注解

@Repository
public class UserDaoImpl implements UserDao {
}

通过注解加扫描配置的id,默认值为类的小驼峰,即类的类名为小写的结果,可以通过在注解后面修改,即通过标识组件的注解的value属性值来设置自定义的id

(2)基于注解方式实现属性注入

(1)@Autowired:根据属性类型进行自动装配

能够标识的位置:
1.成员变量
2.set方法:如果成员变量有对应的set方法,直接标到set方法上也能注入
3.为当前成员变量赋值的有参构造方法上
原理:
默认使用byType的方式,在ioc容器中通过类型匹配某个bean为属性赋值

第一步 把 service 和 dao 对象创建,在 service 和 dao 类添加创建对象注解

第二步 在 service 注入 dao 对象,在 service 类添加 dao 类型属性,在属性上面使用注解

@Controller
public class UserController {
    @Autowired
    private UserService userService;
    public void save(){
        userService.save();
    }
}

(2)@Qualifier:根据名称进行注入
这个@Qualifier 注解的使用,和上面@Autowired 一起使用

(3)@Resource:可以根据类型注入,可以根据名称注入

3.完全注解开发

(1)创建配置类,替代 xml 配置文件

举例:

@Configuration //将当前类作为配置类 @ComponentScan(basePackages = {"com.li.spring"}) //开启组件扫描 public class SpringConfig { } 

(2)编写测试类

举例:

public void TestTwo(){ //加载配置类 ApplicationContext contextTwo =new AnnotationConfigApplicationContext(SpringConfig.class); UserController bean = contextTwo.getBean(UserController.class); bean.save(); } 

三,Bean 作用域

作用域:在Spring 里面,设置创建 bean 实例是单实例还是多实例
在 Spring 里面,默认情况下,bean 是单实例对象

设置单实例还是多实例:设置 scope 值:singleton 或 prototype

singleton 和 prototype 区别
第一 :
singleton 单实例,prototype 多实例
第二 :
设置 scope 值是 singleton 时候,加载 spring 配置文件时候就会创建 单实例对象
设置 scope 值是 prototype 时候,不是在加载 spring 配置文件时候创建 对象,在调用getBean 方法时候创建多实例对象

代码演示:

单例

在这里插入图片描述

多例

在这里插入图片描述

四,Bean 生命周期

1、生命周期
(1)从对象创建到对象销毁的过程
2、bean 生命周期
(1)通过构造器创建 bean 实例(无参数构造)
(2)为 bean 的属性设置值和对其他 bean 引用(调用 set 方法)
(3)调用 bean 的初始化的方法(需要进行配置初始化的方法)
(4)bean 可以使用了(对象获取到了)
(5)当容器关闭时候,调用 bean 的销毁的方法(需要进行配置销毁的方法)

五,Spring中Bean的类型

1、Spring 有两种类型 bean,一种普通 bean,另外一种工厂 bean(FactoryBean)
2、普通 bean:在配置文件中定义 bean 类型就是返回类型
3、工厂 bean:在配置文件定义 bean 类型可以和返回类型不一样

第一步 创建类,让这个类作为工厂 bean,实现接口 FactoryBean
第二步 实现接口里面的方法,在实现的方法中定义返回的 bean 类型

package com.li.factor; import com.li.dao.pojo.user; import org.springframework.beans.factory.FactoryBean; public class userFactor implements FactoryBean<user> { @Override public user getObject() throws Exception { return new user(); } @Override public Class<?> getObjectType() { return user.class; } } 
<?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 http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean class="com.li.factor.userFactor"> </bean> </beans> 

在这里插入图片描述

相关文章
|
1月前
|
XML 缓存 Java
Spring源码之 Bean 的循环依赖
循环依赖是 Spring 中经典问题之一,那么到底什么是循环依赖?简单说就是对象之间相互引用, 如下图所示: 代码层面上很好理解,在 bean 创建过程中 class A 和 class B 又经历了怎样的过程呢? 可以看出形成了一个闭环,如果想解决这个问题,那么在属性填充时要保证不二次创建 A对象 的步骤,也就是必须保证从容器中能够直接获取到 B。 一、复现循环依赖问题 Spring 中默认允许循环依赖的存在,但在 Spring Boot 2.6.x 版本开始默认禁用了循环依赖 1. 基于xml复现循环依赖 定义实体 Bean java复制代码public class A {
|
1月前
|
Java 测试技术 数据库连接
【Spring源码解读!底层原理高级进阶】【下】探寻Spring内部:BeanFactory和ApplicationContext实现原理揭秘✨
【Spring源码解读!底层原理高级进阶】【下】探寻Spring内部:BeanFactory和ApplicationContext实现原理揭秘✨
|
1天前
|
XML 人工智能 Java
Spring Bean名称生成规则(含源码解析、自定义Spring Bean名称方式)
Spring Bean名称生成规则(含源码解析、自定义Spring Bean名称方式)
|
8天前
|
Java 关系型数据库 MySQL
一套java+ spring boot与vue+ mysql技术开发的UWB高精度工厂人员定位全套系统源码有应用案例
UWB (ULTRA WIDE BAND, UWB) 技术是一种无线载波通讯技术,它不采用正弦载波,而是利用纳秒级的非正弦波窄脉冲传输数据,因此其所占的频谱范围很宽。一套UWB精确定位系统,最高定位精度可达10cm,具有高精度,高动态,高容量,低功耗的应用。
一套java+ spring boot与vue+ mysql技术开发的UWB高精度工厂人员定位全套系统源码有应用案例
|
23天前
|
XML Java 数据格式
Spring(一)IOC小案例
Spring(一)IOC小案例
|
1月前
|
Java Spring
使用spring实现邮件的发送(含测试,源码,注释)
使用spring实现邮件的发送(含测试,源码,注释)
7 0
|
1月前
|
存储 Java 关系型数据库
Spring Batch学习记录及示例项目代码
Spring Batch学习记录及示例项目代码
|
1月前
|
XML Java 数据格式
Spring 的奇幻起源:从 IoC 容器到 Bean 的魔法世界 (下)
Spring 的奇幻起源:从 IoC 容器到 Bean 的魔法世界
|
1月前
|
XML Java 数据格式
Spring 的奇幻起源:从 IoC 容器到 Bean 的魔法世界 (上)
Spring 的奇幻起源:从 IoC 容器到 Bean 的魔法世界 (上)
|
1月前
|
Java Spring 容器
【Spring源码】单例创建期间进行同步可能会导致死锁?
通过这个标题我们就可以思考本次的阅读线索了,看起来可以学到不少东西。1. 旧代码的死锁是怎么产生的。2. 贡献者通过改变什么来解决本次PR的问题呢?而阅读线索2的答案也显而易见,就是上文提到的通过后台线程来创建Micrometer单例...
42 3