【Spring】IoC容器 控制反转 与 DI依赖注入 XML实现版本 第二期

文章目录

  • [基于 XML 配置方式组件管理](#基于 XML 配置方式组件管理)
  • [前置 准备项目](#前置 准备项目)
  • [一、 组件(Bean)信息声明配置(IoC):](#一、 组件(Bean)信息声明配置(IoC):)
    • [1.1 基于无参构造](#1.1 基于无参构造)
    • [1.2 基于静态 工厂方法实例化](#1.2 基于静态 工厂方法实例化)
    • [1.3 基于非静态 工厂方法实例化](#1.3 基于非静态 工厂方法实例化)
  • [二、 组件(Bean)依赖注入配置(DI):](#二、 组件(Bean)依赖注入配置(DI):)
    • [1. 1 基于构造函数的依赖注入(单个构造参数)](#1. 1 基于构造函数的依赖注入(单个构造参数))
    • [1.2 基于构造函数的依赖注入(多构造参数解析)](#1.2 基于构造函数的依赖注入(多构造参数解析))
    • [* 1.3 基于Setter方法依赖注入](#* 1.3 基于Setter方法依赖注入)
  • [三、 IoC容器创建和使用:](#三、 IoC容器创建和使用:)
    • [1. 1 容器实例化](#1. 1 容器实例化)
    • [1.2 Bean对象读取](#1.2 Bean对象读取)
  • [四、 高级特性: 组件(Bean)作用域和周期方法配置:](#四、 高级特性: 组件(Bean)作用域和周期方法配置:)
    • [1. 1 组件周期](#1. 1 组件周期)
    • [1.2 组件作用域](#1.2 组件作用域)
  • [五、 高级特性:FactoryBean特性和使用:](#五、 高级特性:FactoryBean特性和使用:)
    • [1. 1 FactoryBean简介](#1. 1 FactoryBean简介)
    • [1.2 FactoryBean应用](#1.2 FactoryBean应用)
    • [1.3 FactoryBean和BeanFactory区别](#1.3 FactoryBean和BeanFactory区别)
  • 六、基于XML方式整合三层架构组件
    • [1.1 需求分析](#1.1 需求分析)
    • [1.2 项目准备](#1.2 项目准备)
    • [1.3 配置文件](#1.3 配置文件)
    • [1.4 测试 数据库 增删改查](#1.4 测试 数据库 增删改查)
    • [1.5 三层结构 搭建 开始](#1.5 三层结构 搭建 开始)
  • 总结
    • [XMLIoC方式问题总结 :](#XMLIoC方式问题总结 :)

基于 XML 配置方式组件管理

前置 准备项目

  • 创建项目工程
  • 导入springioc依赖

    父模块 导入子都需要的依赖,子模块继承:
    pom.xml
xml 复制代码
    <dependencies>
        <!--spring context依赖-->
        <!--当你引入Spring Context依赖之后,表示将Spring的基础依赖引入了-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>6.0.6</version>
        </dependency>
        <!--junit5测试-->
        <dependency>
            <groupId>org.junit.jupiter</groupId>
            <artifactId>junit-jupiter-api</artifactId>
            <version>5.3.1</version>
        </dependency>
    </dependencies>

一、 组件(Bean)信息声明配置(IoC):

1.1 基于无参构造

  1. 创建无参构造 类
  2. 创建spring配置文件
  3. 配置文件中 创建bean
-   bean标签:通过配置bean标签告诉IOC容器需要创建对象的组件信息
-   id属性:bean的唯一标识,方便后期获取Bean!
-   class属性:组件类的全限定符!
-   注意:要求当前组件类必须包含无参数构造函数!

1.2 基于静态 工厂方法实例化

factory-method: 指定静态工厂方法,注意,该方法必须是static方法。

1.3 基于非静态 工厂方法实例化

xml 复制代码
<!-- 将工厂类进行ioc配置 -->
    <bean id="notStaticMethod" class="com.doug.ClientService"></bean>

<!-- 根据工厂对象的实例工厂方法进行实例化组件对象 -->
    <bean id="clientService" factory-bean="notStaticMethod" factory-method="createInstance"></bean>
  • factory-bean属性:指定当前容器中工厂Bean 的名称。
  • factory-method: 指定实例工厂方法名。注意,实例方法必须是非static的!

二、 组件(Bean)依赖注入配置(DI):

1. 1 基于构造函数的依赖注入(单个构造参数)

1.2 基于构造函数的依赖注入(多构造参数解析)

-   constructor-arg标签:指定构造参数和对应的值
-   constructor-arg标签:name属性指定参数名、index属性指定参数角标、value属性指定普通属性值

* 1.3 基于Setter方法依赖注入

  • property标签: 可以给setter方法对应的属性赋值
  • property 标签: name属性代表set方法标识 、ref代表引用bean的标识id、value属性代表基本属性值
    • name 本质是去找方法(set方法),去掉set的

三、 IoC容器创建和使用:

1. 1 容器实例化

读取配置文件,实例化Bean对象

java 复制代码
// 方法一:
//参数:String...locations 传入一个或者多个配置文件
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring-bean-01.xml","spring-bean-02.xml");
 
 //方法二:
 // 先实例化,再指定配置文件,最后刷新容器触发Bean实例化
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext();
//设置配置配置文件,方法参数为可变参数,可以设置一个或者多个配置
context.setConfigLocations("spring-bean-01.xml","spring-bean-02.xml");
//后配置的文件,需要调用refresh方法,触发刷新配置
context.refresh();

1.2 Bean对象读取

java 复制代码
//方式1: 根据id获取
//没有指定类型,返回为Object,需要类型转化!
HappyComponent happyComponent = 
        (HappyComponent) applicationContext.getBean("bean的id标识");
        
//使用组件对象        
happyComponent.doWork();

//方式2: 根据类型获取
//根据类型获取,但是要求,同类型(当前类,或者之类,或者接口的实现类)只能有一个对象交给IoC容器管理
//配置两个或者以上出现: org.springframework.beans.factory.NoUniqueBeanDefinitionException 问题
HappyComponent happyComponent = applicationContext.getBean(HappyComponent.class);
happyComponent.doWork();

//方式3: 根据id和类型获取 !
// getBean("bean的id标识",class)
HelloComponent helloComponent = applicationContext.getBean("helloComponent", HelloComponent.class);
helloComponent.doWork();
  • 根据类型 来获取bean时,在满足bean唯一性的前提下,其实只是看:
    • 对象 instanceof 指定的类型』的返回结果,
  • 只要返回的是true就可以认定为和类型匹配,能够获取到。
  • 根据类型获取,同一个类型,在IoC容器中只能有一个bean
  • 存在多个同类型bean 会报错 (见上代码!)
  • IOC的配置一定是实现类,但可以根据接口类型获取值。
    • interfaceA happyComponent = applicationContext.getBean(interfaceA.class);
    • happyComponent.doWork();
    • 根据实现类的父接口 调用它实现类的方法

四、 高级特性: 组件(Bean)作用域和周期方法配置:

1. 1 组件周期

IoC容器实例化和销毁组件对象

  1. 周期方法声明

  2. 周期方法配置bean

java 复制代码
<bean id="beanPeriod" class="com.doug.BeanPeriod" init-method="Init" destroy-method="destroy"/>
  1. 测试
java 复制代码
        ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring-bean-01.xml");
        // 正常关闭 , 没调用这个方法,就只执行初始化,然后直接寄(非正常死亡)
        applicationContext.close();

1.2 组件作用域

默认就是单例模式,

多例模式:组件对象 getBean 一次就会创建一个组件对象

五、 高级特性:FactoryBean特性和使用:

1. 1 FactoryBean简介

  • BeanFactory IoC容器最大的接口 IoC容器
  • FactoryBean 标准化组件工厂的接口 组件
    FactoryBean用于配置复杂的Bean对象 ,可以将创建过程存储在FactoryBeangetObject方法

FactoryBean<T> 接口提供三种方法:

  • T getObject():

    返回此工厂创建的对象的实例。该返回值会被存储到IoC容器!

  • boolean isSingleton():

    如果此 FactoryBean 返回单例,则返回 true ,否则返回 false 。此方法的默认实现返回 true (注意,lombok插件使用,可能影响效果)。

  • Class<?> getObjectType(): 返回 getObject() 方法返回的对象类型,如果事先不知道类型,则返回 null

1.2 FactoryBean应用

FactoryBean使用场景:

  1. 代理类的创建
  2. 第三方框架整合
  3. 复杂对象实例化等
  • 准备FactoryBean的实现类
  • 配置FactoryBean实现类
xml 复制代码
    <!--
     id -> getObject方法返回的对象标识
     class -> factoryBean标准化工厂类
        -->
    <bean id="javaBean" class="com.doug.ioc02.JavaBeanFactoryBean">
        <property name="value" value="道格维克"/>
    </bean>
  • 测试

1.3 FactoryBean和BeanFactory区别

  • FactoryBean

    • 是 Spring 中一种特殊的 bean,可以在 getObject() 工厂方法自定义的逻辑创建Bean!
    • 是一种能够生产其他 Bean 的 Bean
    • FactoryBean 在容器启动时被创建,而在实际使用时则是通过调用 getObject() 方法来得到其所生产的 Bean
    • FactoryBean 可以自定义任何所需的初始化逻辑,生产出一些定制化的 bean
    • 一般情况下,整合第三方框架,都是通过定义FactoryBean实现
  • BeanFactory

    • Spring 框架的基础,其作为一个顶级接口定义了容器的基本行为
      • 例如管理 bean 的生命周期、配置文件的加载和解析、bean 的装配和依赖注入等
  • BeanFactory 接口提供了访问 bean 的方式
    总的来说 , FactoryBean 和 BeanFactory 的区别 :

  • 主要在于FactoryBean 是用于创建 bean 的接口,它提供了更加灵活的初始化定制功能,

  • 而 BeanFactory是用于管理 bean 的框架基础接口,提供了基本的容器功能和 bean 生命周期管理。

六、基于XML方式整合三层架构组件

1.1 需求分析

搭建一个三层架构案例,模拟查询全部学生(学生表)信息,持久层使用JdbcTemplate和Druid技术,使用XML方式进行组件管理!

1.2 项目准备

  • 数据库:
sql 复制代码
create database studb;

use studb;

CREATE TABLE students (
  id INT PRIMARY KEY AUTO_INCREMENT,
  name VARCHAR(50) NOT NULL,
  gender VARCHAR(10) NOT NULL,
  age INT,
  class VARCHAR(50)
);

INSERT INTO students (id, name, gender, age, class)
VALUES
  (1, '张三', '男', 20, '高中一班'),
  (2, '李四', '男', 19, '高中二班'),
  (3, '王五', '女', 18, '高中一班'),
  (4, '赵六', '女', 20, '高中三班'),
  (5, '刘七', '男', 19, '高中二班'),
  (6, '陈八', '女', 18, '高中一班'),
  (7, '杨九', '男', 20, '高中三班'),
  (8, '吴十', '男', 19, '高中二班');
  • 项目创建:
    spring-ioc-xml-practice-02
  • 依赖导入:
xml 复制代码
<dependencies>
      <!--spring context依赖-->
      <!--当你引入SpringContext依赖之后,表示将Spring的基础依赖引入了-->
      <dependency>
          <groupId>org.springframework</groupId>
          <artifactId>spring-context</artifactId>
          <version>6.0.6</version>
      </dependency>

      <!-- 数据库驱动和连接池-->
      <dependency>
          <groupId>mysql</groupId>
          <artifactId>mysql-connector-java</artifactId>
          <version>8.0.25</version>
      </dependency>

      <dependency>
          <groupId>com.alibaba</groupId>
          <artifactId>druid</artifactId>
          <version>1.2.8</version>
      </dependency>

      <!-- spring-jdbc -->
      <dependency>
          <groupId>org.springframework</groupId>
          <artifactId>spring-jdbc</artifactId>
          <version>6.0.6</version>
      </dependency>

</dependencies> 
  • 实体类准备:
java 复制代码
package com.doug.ioc;

public class Student {
    private Integer id;
    private String name;
    private String gender;
    private Integer age;
    private String classes;

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getGender() {
        return gender;
    }

    public void setGender(String gender) {
        this.gender = gender;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    public String getClasses() {
        return classes;
    }

    public void setClasses(String classes) {
        this.classes = classes;
    }

    @Override
    public String toString() {
        return "Student{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", gender='" + gender + '\'' +
                ", age=" + age +
                ", classes='" + classes + '\'' +
                '}';
    }
}

1.3 配置文件

从JAVA写法 改成 bean创建对象

1.4 测试 数据库 增删改查

java 复制代码
    @Test
    public void testForIoC(){
        // 1. 创建IOC容器
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring-ioc01.xml");
        // 2. 获取jdbc组件
        JdbcTemplate jdbcTemplate = applicationContext.getBean(JdbcTemplate.class);
        // 3. 进行数据库的crud操作
        // 3.1 插入 删除 修改 DML update
        String sql = "insert into students(name,gender,age,class) values(?,?,?,?)";
        /*
        * 参数一:sql语句
        * 参数二: 传入占位符?的值 按顺序
        * 返回值:int 影响行数
        * */
        int rows = jdbcTemplate.update(sql, "道格", "男", 25, "十年十班");
        System.out.println(rows);

        // 3.2 查询单条信息
        // 按照 ID 查询一条学生信息 返回实体对象
        sql = "select * from students where id = ?";
        /*
        * 参数一:sql语句
        * 参数二:RowMapper 列名和属性名的映射器接口
        * 参数三:Object...param 占位符 (这里是id数字)
        * 返回值: rowMapper指定的对象  结果返回实体类
        * */
        Student queryStudent = jdbcTemplate.queryForObject(sql, (rs, rowNum) -> {
            // 返回的结果集对象
            // rs 结果集
            // rowNum 行数
            // rs 结果集中获取列的值 赋值给实体类对象即可
            Student student = new Student();
            student.setId(rs.getInt("id"));
            student.setName(rs.getString("name"));
            student.setGender(rs.getString("gender"));
            student.setAge(rs.getInt("age"));
            student.setClasses(rs.getString("class"));
            return student;
        }, 1);
        System.out.println(queryStudent);

        // 3.3 查询所有学生的信息
        sql = "select id,name,gender,age,class as classes from students";
        // BeanPropertyRowMapper 帮助我们自动映射列和属性值!要求列名和属性名一致!不一致需要起别名!
        List<Student> studentList = jdbcTemplate.query(sql, new BeanPropertyRowMapper<Student>(Student.class));
        System.out.println(studentList);

结果:

1.5 三层结构 搭建 开始

  • dao

  • service

  • controller
  • xml 配置文件
xml 复制代码
    <context:property-placeholder location="classpath:jdbc.properties"/>

    <bean id="druidDataSource" class="com.alibaba.druid.pool.DruidDataSource">
        <property name="url" value="${doug.url}"/>
        <property name="driverClassName" value="${doug.driver}"/>
        <property name="username" value="${doug.username}"/>
        <property name="password" value="${doug.password}"/>
    </bean>

    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="druidDataSource"/>
    </bean>

    <bean id="stuDao" class="com.doug.dao.impl.StuDaoImpl">
        <property name="jdbcTemplate" ref="jdbcTemplate"/>
    </bean>

    <bean id="stuService" class="com.doug.service.impl.StuServiceImpl">
        <property name="stuDao" ref="stuDao"/>
    </bean>

    <bean id="stuController" class="com.doug.controller.StuController">
        <property name="stuService" ref="stuService"/>
    </bean>
  • 测试
java 复制代码
    @Test
    public void testStuControllerMethod(){
        ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring-02.xml");
        StuController stuController = applicationContext.getBean(StuController.class);
        stuController.findAllStu();
    }

总结

lambada

XMLIoC方式问题总结 :

  1. 注入的属性必须添加setter方法、代码结构乱!
  2. 配置文件和Java代码分离、编写不是很方便!
  3. XML配置文件解析效率低
相关推荐
bug菌22 分钟前
Java GUI编程进阶:多线程与并发处理的实战指南
java·后端·java ee
程序猿小D35 分钟前
第二百六十九节 JPA教程 - JPA查询OrderBy两个属性示例
java·开发语言·数据库·windows·jpa
极客先躯1 小时前
高级java每日一道面试题-2024年10月3日-分布式篇-分布式系统中的容错策略都有哪些?
java·分布式·版本控制·共识算法·超时重试·心跳检测·容错策略
夜月行者2 小时前
如何使用ssm实现基于SSM的宠物服务平台的设计与实现+vue
java·后端·ssm
程序猿小D2 小时前
第二百六十七节 JPA教程 - JPA查询AND条件示例
java·开发语言·前端·数据库·windows·python·jpa
Yvemil72 小时前
RabbitMQ 入门到精通指南
开发语言·后端·ruby
sdg_advance2 小时前
Spring Cloud之OpenFeign的具体实践
后端·spring cloud·openfeign
潘多编程2 小时前
Java中的状态机实现:使用Spring State Machine管理复杂状态流转
java·开发语言·spring
_阿伟_2 小时前
SpringMVC
java·spring
代码在改了3 小时前
springboot厨房达人美食分享平台(源码+文档+调试+答疑)
java·spring boot