0%

Spring5.x学习笔记3——反转控制与依赖注入、Spring工厂创建复杂对象3种方式

Spring5.x学习笔记3——反转控制与依赖注入、Spring工厂创建复杂对象3种方式

反转控制 与 依赖注入

反转控制(IOC Inverse of Control)

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

  • 控制:对于成员变量赋值的控制权;
  • 反转控制:把对于成员变量赋值的控制权,从代码中转移(反转)到 Spring ⼯⼚和配置⽂件中完成。
  • 好处:解耦合;
  • 底层实现:工厂设计模式;
    在这里插入图片描述

依赖注入 (Dependency Injection - DI)

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

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

什么是复杂对象

在这里插入图片描述
简单对象:可以直接通过 new 构造方法创建的对象;

1
2
3
4
5
6
UserService
UserDAO
Customer
Person
......
12345

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

1
2
3
4
Connection
SqlSessionFactory
......
123

FactoryBean 接口

  • 实现 FactoryBean 接口

    :实现三个方法:

    • getObject():用于书写创建复杂对象时的代码。
    • getObjectType():返回创建的复杂对象的类型。
    • isSingleton:用于决定是否单例。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
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; // 只创建一次这种类型的复杂对象
}
}
12345678910111213141516171819202122
  • Spring 配置文件的配置:如果 class 中指定的类型是 FactoryBean 接⼝的实现类,那么通过 id 值获得的是这个类所创建的复杂对象。
    比如下面 class 指定的是 ConnectionFactoryBean,获得的是 Connection 对象。
1
2
3
<!--class 指定了 ConnectionFactoryBean, 获得的是该类创建的复杂对象 Connection -->
<bean id="conn" class="com.yusael.factorybean.ConnectionFactoryBean"/>
12

FactoryBean 细节

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

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

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


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

1
2
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.
1

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


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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
@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;
}
}
12345678910111213141516171819202122
<!--体会依赖注入, 好处: 解耦合, 今后要修改连接数据库的信息只需要修改配置文件, 无需改动代码-->
<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>
1234567

FactoryBean 实现原理[简易版]

原理:接口回调

问题:

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

Spring 内部运行流程:

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

在这里插入图片描述
FactoryBean 总结:Spring 中用于创建复杂对象的,也是 Spring 提供的,后续 Spring 整合其他框架时会大量应用FactoryBean 方式。

实例工厂

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

[开发步骤]:

  • ConnectionFactory 类
1
2
3
4
5
6
7
8
9
10
11
12
13
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;
}
}
123456789101112
  • 配置文件:
1
2
3
4
5
6
<!--实例工厂-->
<!-- 先创建出工厂实例 -->
<bean id="connFactory" class="com.yusael.factorybean.ConnectionFactory"/>
<!-- 通过工厂实例里的方法创建复杂对象 -->
<bean id="conn" factory-bean="connFactory" factory-method="getConnection"/>
12345

静态工厂

[开发步骤]:

  • StaticConnectionFactory 类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
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;
}
}
12345678910111213
  • 配置文件:
1
2
3
<!--静态工厂-->
<bean id="conn" class="com.yusael.factorybean.StaticFactoryBean" factory-method="getConnection"/>
12

Spring工厂创建对象的总结

在这里插入图片描述
在这里插入图片描述

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

控制简单对象的创建次数

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

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

控制复杂对象的创建次数

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

1
2
3
4
5
6
7
8
public class xxxFactoryBean implements FactoryBean {
public boolean isSingleton() {
return true; // 只会创建⼀次
// return false; // 每⼀次都会创建新的
}
// 省略其余实现方法......
}
1234567

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

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

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

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

  • 重量级的、可以被共用的、线程安全的…
1
2
3
4
5
SqlSessionFactory
DAO
Service
......
1234

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

  • 不能被共用的,线程不安全的…
1
2
3
4
Connection
SqlSession | Session
Struts2 - Action
......
-------------本文结束感谢您的阅读-------------