(一) IOC原型
- 以往的实现步骤
- XXX Dao
- XXX DaoImpl
- XXX Service
- XXX ServiceImpl
- XXX Servlet
存在问题
(1)当Service层调用Dao层时,Dao层会有许多不同的实现类,是根据不同业务而实现的,那么Service层会在调用时出现:选择实现类的问题。
(2)假如Service的实现类,来选择某一个Dao层实现类,由ServiceImpl来创建对应的DaoImpl实例对象,那么ServiceImpl就只能有一个具体的Dao层实现类。
(3)可是当需要使用其他Dao层实现类,我们需要去改变ServiceImpl中的DaoImpl对象
(4)这样程序的耦合性高
控制反转(思想)
假如我们Service层不再实例化一个具体的DaoImpl对象,即程序员不用根据不同的需求而跑到Service层选择不同的DaoImpl类而创建对象
将对象的创建交给更上一层,调用Service层的 Servlet层去确定要使用哪一个DaoImpl类,直接创建并且传入给Service层即可
程序对象创建的主动权,交给了用户。至此,程序员不用管理对象的创建,系统的耦合性降低
-
左边为:业务层决定调用的对象;右边为:用户层决定让业务层去调用某个对象
- 控制反转(思想)代码演示:
- 通过Set注入一个UserDao对象,程序不再具有主动性,而是变成被动的接收对象
public class UserServiceImpl implements UserService {
//引入Dao层
private UserDao userDao = null;
//由创建该对象的实例对象决定使用那一个UserDao
public UserServiceImpl(UserDao userDao) {
this.userDao = userDao;
}
//通过Set注入一个UserDao对象,程序不再具有主动性,而是变成被动的接收对象
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
public void getUser() {
userDao.getUser();
}
}
(二) IOC本质
控制反转IoC(Inversion of Control),是一种设计思想,DI(依赖注入)是实现IoC的一种方法。
- 没有IoC的程序中,我们使用面向对象编程,对象的创建与对象间的依赖关系完全硬编码在程序中,对象的创建由程序自己控制
- 控制反转后,将对象的创建转移给第三方:获得依赖对象的方式反转了
采用XML方式配置Bean的时候,Bean的定义信息是和实现分离的,而采用注解的方式可以把两者合为一体,Bean的定义信息直接以注解的形式定义在实现类中,从而达到了零配置的目的。
控制反转是一种通过描述(XML或注解)并通过第三方去生产或获取特定对象的方式。在Spring中实现控制反转的是IoC容器,其实现方法是依赖注入(Dependency Injection,DI)
(三)HelloSpring
- Hello.java
- 注意:要有无参的构造方法以及参数的set方法
public class Hello {
String str = null;
public String getStr() {
return str;
}
public void setStr(String str) {
this.str = str;
}
@Override
public String toString() {
return "Hello{" +
"str='" + str + '\'' +
'}';
}
}
- 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
http://www.springframework.org/schema/beans/spring-beans.xsd">
<!--
类型 变量名 = new 类型();
Hello hello = new Hello();
id = 变量名
class = 全类名
property 相当于给对象中的属性设置一个值
-->
<!--使用Spring来创建对象,在Spring这些都称为Bean-->
<bean id="hello" class="com.vigil.pojo.Hello">
<property name="str" value="Spring"></property>
</bean>
</beans>
- Test.java
import com.vigil.pojo.Hello;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Test {
public static void main(String[] args) {
//获取Spring的上下文对象,解析beans.xml文件,生成管理相应的Bean对象
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
//对象都由Spring管理,使用的时候直接取出来使用
//getBean:参数即为spring配置文件中的bean的id
Hello hello = (Hello) context.getBean("hello");
System.out.println(hello);
}
}
这一系列的过程就叫控制反转
-
控制:控制对象的创建
- 传统应用程序的对象是由程序本身控制创建
- 使用Spring后,对象是由Spring来创建
反转:程序本身不创建对象,而变为被动的接收对象
依赖注入:利用set方法来进行注入
IOC是一种编程思想,是由主动的编程编程被动的接收
利用Spring,不用在程序中去改动,要实现不同的操作,只需要在xml配置文件中进行修改,所谓IoC:对象由Spring创建、管理、装配
(四)对User实现的修改
- UserServiceImpl
public class UserServiceImpl implements UserService {
//引入Dao层
private UserDao userDao = null;
// public UserServiceImpl(UserDao userDao) {
// this.userDao = userDao;
// }
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
public void getUser() {
userDao.getUser();
}
}
- beans.xml(ApplicationContext)
<?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="Mysql" class="com.vigil.dao.UserDaoMysqlImpl"></bean>
<bean id="SqlServer" class="com.vigil.dao.UserDaoSqlServerImpl"></bean>
<!--
ref:引用Spring容器中创建好的对象
value:具体的值,是基本数据类型
-->
<bean id="UserService" class="com.vigil.service.UserServiceImpl">
<property name="userDao" ref="Mysql"></property>
</bean>
</beans>
- Test.java
public class Test {
public static void main(String[] args) {
// UserService userService = new UserServiceImpl(new UserDaoSqlServerImpl());
// userService.getUser();
//获取ApplicationContext:拿到Spring容器
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
//利用容器获得对象
UserService userService = (UserService) context.getBean("UserService");
userService.getUser();
}
}