Spring5【二】IoC 理论推导

2、IoC 理论推导

2.1 IoC 思想的理解

传统方法:

  1. 新建一个 Maven 工程,用原来的方式写一段代码

  2. UserDao 接口

public interface UserDao {
    void getUser();
}
  1. 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 获取用户数据");
    }
}
  1. UserService 业务接口
public interface UserService {
    void getUser();
}
  1. 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();
    }
}
  1. 测试类
public class MyTest {
    public static void main(String[] args) {
        // 用户实际调用的是业务层,dao 层他们不需要接触
        UserService userService = new UserServiceImpl();
        userService.getUser();
    }
}
  1. 调试

    • 如果使用实现类 UserDaoImpl,在业务层调用 private UserDao userDao = new UserDaoImpl();
    • 如果要使用 UserDaoMysqlImpl,需要在业务层修改调用,变为 private UserDao userDao = new UserDaoMysqlImpl();
    • 同理,如果要使用UserDaoOracleImpl,则将业务层的调用修改为private UserDao userDao = new UserDaoOracleImpl();
  2. 总结

    在这个业务操作中中,用户的需求可能会影响我们原来的代码,我们需要根据用户的需求去修改源代码(业务层代码),如果程序代码量十分大,修改一次的成本代价十分昂贵。

改进:

  1. 在 UserServiceImpl 业务实现类中,使用一个 set 方法传入一个实现类对应的接口,在客户端去传入具体的实现类,这已经发生了革命性变化
public class UserServiceImpl implements UserService {

    private UserDao userDao;

    // 利用 set 进行动态实现值的注入
    public void setUserDao(UserDao userDao) {
        this.userDao = userDao;
    }
    
    public void getUser() {
        userDao.getUser();
    }
}
  1. 测试类:
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();
    }
}
  1. 调试

    • 如果使用实现类 UserDaoImpl,在客户端调用 set 方法传入接口的实现类对象 new UserDaoImpl()
    • 如果要使用 UserDaoMysqlImpl,在客户端调用 set 方法传入 new UserDaoMysqlImpl()
    • 同理,如果要使用UserDaoOracleImpl,在客户端调用 set 方法传入new UserDaoOracleImpl()
  2. 总结

    在需要用到某个实现类的地方,我们不直接去实现它,而是留出一个接口,利用 set() 方法,让客户去修改。即客户需求改变时,不用去修改源代码(不操作业务层),只需要在客户端传入不同的接口实现类对象即可。

对比分析:

  • 之前,程序是主动创建对象,控制权在程序员手上

  • 使用了 set 注入后,程序不再具有主动性,而是变成了被动的接收对象,程序只负责提供一个接口

    这种思想,从本质上解决了问题,程序员不用再去管理对象的创建,系统耦合性大大降低,可以更加专注于业务的实现上,这是 IOC 原型。

2.2 IoC 本质

控制反转 IoC(Inversion of Control),是一种设计思想,DI(依赖注入)是实现IoC的一种方法,也有人认为 DI 只是 IoC 的另一种说法。没有IoC的程序中 , 我们使用面向对象编程 , 对象的创建与对象间的依赖关系完全硬编码在程序中,对象的创建由程序自己控制,控制反转后将对象的创建转移给第三方,个人认为所谓控制反转就是:获得依赖对象的方式反转了。

IoC 是 Spring 框架的核心内容,使用多种方式完美的实现了 IoC,可以使用 XML 配置,也可以使用注解,新版本的 Spring 也可以零配置实现 IoC。

Spring 容器在初始化时先读取配置文件,根据配置文件或元数据创建与组织对象存入容器中,程序使用时再从 Ioc容器中取出需要的对象。

img

采用 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 包,再编写代码

  1. 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 + '\'' +
                    '}';
        }
    }
    
  2. 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>
    
  3. 测试类

    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 来创建、管理、装配。

posted @ 2020-06-19 21:59  Song-zw  阅读(184)  评论(0编辑  收藏  举报