spring(二)IOC理论
2、IOC理论
2.1 IOC组成理论推导
原来的是实现方式
1.UserDao接口
public interface UserDao {
void getUser();
}
2.UserDaoImpl实现类
public class UserDaoImpl implements UserDao{
public void getUser() {
System.out.println("默认获取用户的数据");
}
}
3.UserService业务接口
public interface UserService {
void getUser();
}
4.UserServiceImpl业务实现类
public class UserServiceImpl implements UserService{
private UserDao userDao = new UserDaoImpl();
public void getUser() {
userDao.getUser();
}
}
在我们之前的业务中,用户的需求可能会影响我们原来的代码,我们需要更具用户的需求去修改源代码!如果程序代码量十分大,修改一次的成本代价十分昂贵!
我们使用一个Set接口实现,已经发生了革命性的变化
private UserDao userDao;
// 利用set进行动态实现值的注入
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
如果想要改变,就需要每次更改UserDao
若将UesrDao使用Set接口实现
public class UserServiceImpl implements UserService{
public class UserServiceImpl implements UserService{
private UserDao userDao;
//利用set进行动态实现值得注入
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
public void getUser() {
userDao.getUser();
}
}
-
之前,程序是主动创建对象!控制权在程序员手上!
-
使用了set注入之后,程序不再有主动性,而是变成了被动的接受对象!
-
这种思想从本质上实现了问题,我们程序员不用再去管理对象的创建了。系统的耦合性大大降低,可以更加专注的在业务的实现上!这是IOC的原型!
2.2 IOC本质
控制反转IOC(Inversion of Control),是一种设计思想,DI(依赖注入)是实现IOC的一种方法, 也有人认为DI只是IOC的另一种说法。没有IOC的程序中,我们使用面向对象编程,对象的创建与对象间的依赖关系完全硬编码在程序中,对象的创建由程序自己控制,控制反转后将对象的创建转移给第三方,个人认为所谓的控制反转就是:获得依赖的方式反转了。
采用XML方式配置Bean的时候,Bean的定义信息是和实现分离的,而采用注解的方式可以把两者合为一体,Bean的定义信息直接以注解的形式定义在实现类中,从而达到了零配置的目的。
控制反转是一种通过描述(xml或注解)并通过第三方去生产或获取特定对象的方式。在spring中实现控制反转的是IOC容器,其实现方法是依赖注入(Dependency Injection,DI)
2.3 HelloSpring
配置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">
<!--使用Spring来创建对象,在Spring这些都成为bean
Java: 类型 变量名 = new 类型()
Hello hello = new Hello()
Spring: id = 变量名
class = new的对象
properties 相当于给对象中的属性设置一个值
-->
<bean id = "hello" class="com.wen.pojo.Hello">
<property name="name" value="HelloSpring"/>
</bean>
</beans>
实体类
public class Hello {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Hello{" +
"name='" + name + '\'' +
'}';
}
}
测试
@Test
public void helloSpring(){
// 获取spring的上下文对象
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
// 对象都在spring中管理了,要使用直接取出来就可以了
Hello hello = (Hello) context.getBean("hello");
System.out.println(hello);
}
会发现,已经不需要手动new对象,对象是在xml文件中配置。或者通俗来讲,不需要改底层代码,而xml文件不算底层代码。
控制反转
- 控制: 谁来控制对象的创建,传统应用程序的对象是由程序本身控制创建的,使用Spring后,对象是由Spring来创建的
- 反转: 程序本身不创建对象,而变成被动的接收对象。
- 依赖注入: 就是利用set方法来进行注入的
- IOC是一种编程思想,由主动的编程变为被动的接收,所谓的IOC,即对象由Spring来创建,管理,装配
2.4 IOC创建对象方式
- 默认使用无参构造创建对象
- 使用有参构造创建对象的三种方式
实体类(包含有参构造器)
public class User {
private String name;
private int age;
public User(String name, int age) {this.name = name;this.age = age;}
public String getName() {return name;}
public void setName(String name) {this.name = name;}
public int getAge() {return age;}
public void setAge(int age) {this.age = age;}
}
方式一:变量名称赋值 Constructor argument name
<bean id="user" class="com.wen.pojo.User">
<constructor-arg name="name" value="wendaidai"/>
<constructor-arg name="age" value="18"/>
</bean>
方式二:Constructor argument index
方式三: Constructor argument type matching
在获取spring的上下文对象( new ClassPathXmlApplicationContext(“beans.xml”); )时,spring容器中的所有的对象就已经被创建了。