反转控制与依赖注入

  • 反转控制 与 依赖注入
  • 反转控制(IOC Inverse of Control)
  • 依赖注入 (Dependency Injection - DI)
  • Spring工厂创建复杂对象(3种方式)
  • 什么是复杂对象
  • 1. FactoryBean 接口
  • 开发步骤
  • FactoryBean 细节
  • FactoryBean 实现原理[简易版]
  • 2. 实例工厂
  • 3. 静态工厂
  • Spring工厂创建对象的总结
  • 控制 Spring 工厂创建对象的次数
  • 1. 控制简单对象的创建次数 - scope
  • 2. 控制复杂对象的创建次数
  • 为什么要控制对象的创建次数?



反转控制 与 依赖注入

反转控制(IOC Inverse of Control)

反转控制(IOC Inverse of Control),也称为 转移控制

  • 控制:对于成员变量赋值的控制权;
  • 反转控制:把对于成员变量赋值的控制权,从代码中转移(反转)到 Spring 工厂和配置文件中完成。
  • 好处:解耦合;
  • 底层实现:工厂设计模式;

依赖注入 (Dependency Injection - DI)

  • 注入:通过 Spring 的工厂及配置文件,为对象(bean,组件)的成员变量赋值;
  • 依赖注入:当⼀个类需要另⼀个类时,就意味着依赖,⼀旦出现依赖,就可以把另⼀个类作为本类的成员变量,最终通过 Spring 配置文件进行注入(赋值)。
  • 好处:解耦合;

Spring工厂创建复杂对象(3种方式)

什么是复杂对象

spring boot 工厂模式 集合类 怎么注入 spring工厂方法注入_配置文件


简单对象:可以直接通过 new 构造方法创建的对象;

UserService
UserDAO
Customer
Person
......

复杂对象:不能直接通过 new 构造方法创建的对象。

Connection
SqlSessionFactory
......

1. FactoryBean 接口

开发步骤

  • 实现 FactoryBean 接口:实现 getObjectgetObjectTypeisSingleton 方法;
  • getObject():用于书写创建复杂对象时的代码。
  • getObjectType():返回创建的复杂对象的类型。
  • isSingleton:用于决定是否单例。
public class ConnectionFactoryBean implements FactoryBean<Connection> {
    // 用于书写创建复杂对象时的代码
    @Override
    public Connection getObject() throws Exception {
        Class.forName("com.mysql.jdbc.Driver");
        Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/spring", "root", "1234");
        return conn;
    }

    // 返回创建的复杂对象的类型
    @Override
    public Class<Connection> getObjectType() {
        return Connection.class;
    }

    // 是否单例
    @Override
    public boolean isSingleton() {
        return false; // 每一次都创建新的复杂对象
        // return true; // 只创建一次这种类型的复杂对象
    }
}
  • Spring 配置文件的配置:如果 class 中指定的类型是 FactoryBean 接口的实现类,那么通过 id 值获得的是这个类所创建的复杂对象。
    比如下面 class 指定的是 ConnectionFactoryBean,获得的是 Connection 对象。
<!--class 指定了 ConnectionFactoryBean, 获得的是该类创建的复杂对象 Connection -->
<bean id="conn" class="com.yusael.factorybean.ConnectionFactoryBean"/>

FactoryBean 细节

如果就想获得 FactoryBean 类型的对象,加个 &ctx.getBean("&conn")

ConnectionFactoryBean cfb = (ConnectionFactoryBean) ctx.getBean("&conn");

isSingleton 方法返回 true 只会创建⼀个复杂对象,返回 false 每⼀次都会创建新的对象;
需要根据这个对象的特点 ,决定是返回 true(SqlSessionFactory) 还是 false(Connection);


mysql 高版本连接创建时,需要制定 SSL 证书,否则会警告;

Sat May 23 23:18:04 CST 2020 WARN: Establishing SSL connection without server's identity verification is not recommended. According to MySQL 5.5.45+, 5.6.26+ and 5.7.6+ requirements SSL connection must be established by default if explicit option isn't set. For compliance with existing applications not using SSL the verifyServerCertificate property is set to 'false'. You need either to explicitly disable SSL by setting useSSL=false, or set useSSL=true and provide truststore for server certificate verification.

解决方案:url = jdbc:mysql://localhost:3306/spring?useSSL=false


依赖注入(DI):把 ConnectionFactoryBean 中依赖的 4 个字符串信息 ,通过配置文件进行注入。

@Getter@Setter // 提供 get set 方法
public class ConnectionFactoryBean implements FactoryBean<Connection> {
	// 将依赖的字符串信息变为成员变量, 利用配置文件进行注入。
    private String driverClassName;
    private String url;
    private String username;
    private String password;
    @Override
    public Connection getObject() throws Exception {
        Class.forName(driverClassName);
        Connection conn = DriverManager.getConnection(url, username, password);
        return conn;
    }
    @Override
    public Class<Connection> getObjectType() {
        return Connection.class;
    }
    @Override
    public boolean isSingleton() {
  		return false;
    }
}
<!--体会依赖注入, 好处: 解耦合, 今后要修改连接数据库的信息只需要修改配置文件, 无需改动代码-->
<bean id="conn" class="com.yusael.factorybean.ConnectionFactoryBean">
    <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
    <property name="url" value="jdbc:mysql://localhost:3306/spring?useSSL=false"/>
    <property name="username" value="root"/>
    <property name="password" value="1234"/>
</bean>

FactoryBean 实现原理[简易版]

原理:接口回调

问题:

  1. 为什么 Spring 规定 FactoryBean 接口实现 getObject()
  2. 为什么 ctx.getBean("conn") 获得的是复杂对象 Connection ⽽非 ConnectionFactoryBean

Spring 内部运行流程:

  1. 配置文件中通过 id conn 获得 ConnectionFactoryBean 类的对象 ,进而通过 instanceof 判断出是 FactoryBean 接口的实现类;
  2. Spring 按照规定 getObject() —> Connection
  3. 返回 Connection

spring boot 工厂模式 集合类 怎么注入 spring工厂方法注入_mysql_02


FactoryBean 总结:Spring 中用于创建复杂对象的⼀种方式,也是 Spring 原生提供的,后续 Spring 整合其他框架时会大量应用 FactoryBean 方式。

2. 实例工厂

  1. 避免 Spring 框架的侵入;
  2. 整合遗留系统;

[开发步骤]:

  • ConnectionFactory 类
public class ConnectionFactory {
    public Connection getConnection() {
        Connection conn = null;
        try {
            Class.forName("com.mysql.jdbc.Driver");
            conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/spring?useSSL=false", "root", "1234");
        } catch (ClassNotFoundException | SQLException e) {
            e.printStackTrace();
        }
        return conn;
    }
}
  • 配置文件:
<!--实例工厂-->
<!-- 先创建出工厂实例 -->
<bean id="connFactory" class="com.yusael.factorybean.ConnectionFactory"/>
 <!-- 通过工厂实例里的方法创建复杂对象 -->
<bean id="conn" factory-bean="connFactory" factory-method="getConnection"/>

3. 静态工厂

[开发步骤]:

  • StaticConnectionFactory 类
public class StaticFactoryBean {
	// 静态方法
    public static Connection getConnection() {
        Connection conn = null;
        try {
            Class.forName("com.mysql.jdbc.Driver");
            conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/spring?useSSL=false", "root", "1234");
        } catch (ClassNotFoundException | SQLException e) {
            e.printStackTrace();
        }
        return conn;
    }
}
  • 配置文件:
<!--静态工厂-->
<bean id="conn" class="com.yusael.factorybean.StaticFactoryBean" factory-method="getConnection"/>

Spring工厂创建对象的总结

spring boot 工厂模式 集合类 怎么注入 spring工厂方法注入_mysql_03


spring boot 工厂模式 集合类 怎么注入 spring工厂方法注入_配置文件_04

控制 Spring 工厂创建对象的次数

1. 控制简单对象的创建次数 - scope

配置文件中进行配置:
singleton:每一个 IoC 容器只会创建⼀次简单对象,默认值;
prototype:每⼀次都会创建新的对象;

<!--控制简单对象创建次数-->
<bean id="scope" scope="singleton" class="com.yusael.scope.Scope"/>

2. 控制复杂对象的创建次数

如果是 FactoryBean 方式创建的复杂对象:

public class xxxFactoryBean implements FactoryBean {
	public boolean isSingleton() {
		return true; // 只会创建⼀次
		// return false; // 每⼀次都会创建新的
	}
	// 省略其余实现方法......
}

如果是实例工厂或者静态工厂,没有 isSingleton ⽅法,与简单对象一样通过 scope 控制。

为什么要控制对象的创建次数?

好处:节省不必要的内存浪费。

什么样的对象只创建⼀次?

  • 重量级的、可以被共用的、线程安全的…
SqlSessionFactory
DAO
Service
......

什么样的对象每⼀次都要创建新的?

  • 不能被共用的,线程不安全的…
Connection
SqlSession | Session
Struts2 - Action
......

spring boot 工厂模式 集合类 怎么注入 spring工厂方法注入_bc_05