Spring5【二】IoC 理论推导
2、IoC 理论推导
2.1 IoC 思想的理解
传统方法:
-
新建一个 Maven 工程,用原来的方式写一段代码
-
UserDao 接口
public interface UserDao {
void getUser();
}
- UserDaoImpl 实现类(有多个 UserDaoMysqlImpl、UserDaoOracleImpl......)
public class UserDaoImpl implements UserDao {
public void getUser() {
System.out.println("默认获取用户的数据");
}
}
public class UserDaoMysqlImpl implements UserDao {
public void getUser() {
System.out.println("mysql 获取用户的数据");
}
}
public class UserDaoOracleImpl implements UserDao {
public void getUser() {
System.out.println("Oracle 获取用户数据");
}
}
- UserService 业务接口
public interface UserService {
void getUser();
}
- UserServiceImpl 业务实现类
public class UserServiceImpl implements UserService {
private UserDao userDao = new UserDaoImpl();
// private UserDao userDao = new UserDaoMysqlImpl();
// private UserDao userDao = new UserDaoOracleImpl();
public void getUser() {
userDao.getUser();
}
}
- 测试类
public class MyTest {
public static void main(String[] args) {
// 用户实际调用的是业务层,dao 层他们不需要接触
UserService userService = new UserServiceImpl();
userService.getUser();
}
}
-
调试
- 如果使用实现类
UserDaoImpl
,在业务层调用private UserDao userDao = new UserDaoImpl();
- 如果要使用
UserDaoMysqlImpl
,需要在业务层修改调用,变为private UserDao userDao = new UserDaoMysqlImpl();
- 同理,如果要使用
UserDaoOracleImpl
,则将业务层的调用修改为private UserDao userDao = new UserDaoOracleImpl();
- 如果使用实现类
-
总结
在这个业务操作中中,用户的需求可能会影响我们原来的代码,我们需要根据用户的需求去修改源代码(业务层代码),如果程序代码量十分大,修改一次的成本代价十分昂贵。
改进:
- 在 UserServiceImpl 业务实现类中,使用一个 set 方法传入一个实现类对应的接口,在客户端去传入具体的实现类,这已经发生了革命性变化
public class UserServiceImpl implements UserService {
private UserDao userDao;
// 利用 set 进行动态实现值的注入
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
public void getUser() {
userDao.getUser();
}
}
- 测试类:
public class MyTest {
public static void main(String[] args) {
// 用户实际调用的是业务层,dao 层他们不需要接触
UserService userService = new UserServiceImpl();
// ((UserServiceImpl) userService).setUserDao(new UserDaoImpl());
((UserServiceImpl) userService).setUserDao(new UserDaoMysqlImpl());
// ((UserServiceImpl) userService).setUserDao(new UserDaoOracleImpl());
userService.getUser();
}
}
-
调试
- 如果使用实现类
UserDaoImpl
,在客户端调用 set 方法传入接口的实现类对象new UserDaoImpl()
- 如果要使用
UserDaoMysqlImpl
,在客户端调用 set 方法传入new UserDaoMysqlImpl()
- 同理,如果要使用
UserDaoOracleImpl
,在客户端调用 set 方法传入new UserDaoOracleImpl()
- 如果使用实现类
-
总结
在需要用到某个实现类的地方,我们不直接去实现它,而是留出一个接口,利用 set() 方法,让客户去修改。即客户需求改变时,不用去修改源代码(不操作业务层),只需要在客户端传入不同的接口实现类对象即可。
对比分析:
-
之前,程序是主动创建对象,控制权在程序员手上
-
使用了 set 注入后,程序不再具有主动性,而是变成了被动的接收对象,程序只负责提供一个接口
这种思想,从本质上解决了问题,程序员不用再去管理对象的创建,系统耦合性大大降低,可以更加专注于业务的实现上,这是 IOC 原型。
2.2 IoC 本质
控制反转 IoC(Inversion of Control),是一种设计思想,DI(依赖注入)是实现IoC的一种方法,也有人认为 DI 只是 IoC 的另一种说法。没有IoC的程序中 , 我们使用面向对象编程 , 对象的创建与对象间的依赖关系完全硬编码在程序中,对象的创建由程序自己控制,控制反转后将对象的创建转移给第三方,个人认为所谓控制反转就是:获得依赖对象的方式反转了。
IoC 是 Spring 框架的核心内容,使用多种方式完美的实现了 IoC,可以使用 XML 配置,也可以使用注解,新版本的 Spring 也可以零配置实现 IoC。
Spring 容器在初始化时先读取配置文件,根据配置文件或元数据创建与组织对象存入容器中,程序使用时再从 Ioc容器中取出需要的对象。
采用 XML 方式配置 Bean 的时候,Bean 的定义信息是和实现分离的,而采用注解的方式可以把两者合为一体,Bean 的定义信息直接以注解的形式定义在实现类中,从而达到了零配置的目的。
控制反转是一种通过描述(XML 或注解)并通过第三方去生产或获取特定对象的方式。在Spring中实现控制反转的是IoC容器,其实现方法是依赖注入(Dependency Injection, DI)。
3、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
https://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="..." class="...">
<!-- collaborators and configuration for this bean go here -->
</bean>
<bean id="..." class="...">
<!-- collaborators and configuration for this bean go here -->
</bean>
<!-- more bean definitions go here -->
</beans>
ApplicationContext context = new ClassPathXmlApplicationContext("services.xml", "daos.xml");
3.1 Hello Stirng 案例
先导入依赖 jar 包,再编写代码
-
Hello 实体类
public class Hello { private String str; public String getStr() { return str; } public void setStr(String str) { this.str = str; } @Override public String toString() { return "Hello{" + "str='" + str + '\'' + '}'; } }
-
Spring 配置文件(头部信息参考官方文档),这里命名为 beans.xml
<?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 https://www.springframework.org/schema/beans/spring-beans.xsd"> <!--使用 Spring 创建对象,在 Spring 中这些都称为 Bean 类型 变量名 = new 类型(); Hello hello = new Hello(); id = 变量名 class = new 的对象 property 相当于给对象中的属性设置一个值 --> <bean id="hello" class="com.song.pojo.Hello"> <property name="str" value="Spring"/> </bean> </beans>
-
测试类
public class MyTest { public static void main(String[] args) { // 获取 Spring 的上下文对象 ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml"); // 我们的对象现在都在 Spring 中管理了,如果要使用,直接取出来就可以 Hello hello = (Hello) context.getBean("hello"); System.out.println(hello.toString()); } }
总结:
- Hello 对象是谁创建的 ? 【hello 对象是由 Spring 创建的】
- Hello 对象的属性是怎么设置的 ? 【hello 对象的属性是由 Spring 容器设置的】
这个过程就叫控制反转(IoC) :
- 控制 : 谁来控制对象的创建 , 传统应用程序的对象是由程序本身控制创建的 , 使用 Spring 后 , 对象是由 Spring 来创建的
- 反转 : 程序本身不创建对象 , 而变成被动的接收对象
- IOC是一种编程思想,由主动的编程变成被动的接收。
依赖注入 : 就是利用 set() 方法来进行注入的。
3.2 案例修改
对 2.1 中的案例进行修改,使用 Spring 容器来管理对象。
新增一个 Spring 配置文件 beans.xml
<?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
https://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="mysqlImpl" class="com.song.dao.UserDaoMysqlImpl"/>
<bean id="oracleImpl" class="com.song.dao.UserDaoOracleImpl"/>
<bean id="userServiceImpl" class="com.song.service.UserServiceImpl">
<!--<property name="userDao" ref="mysqlImpl"/>-->
<property name="userDao" ref="oracleImpl"/>
</bean>
<!--
ref:引用 Spring 容器中创建好的配置文件
value:具体的值,基本数据类型
-->
</beans>
测试类:
public class MyTest1 {
public static void main(String[] args) {
// 获取 ApplicationContext:拿到 Spring 容器
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
// 需要什么对象,直接 get
UserServiceImpl userServiceImpl = (UserServiceImpl)context.getBean("userServiceImpl");
userServiceImpl.getUser();
}
}
要实现不同的操作,我们彻底不用在程序中去改动,只需要在 xml 配置文件中进行修改。
所谓 IoC:对象由 Spring 来创建、管理、装配。