Spring - IOC入门(01)

IOC

IOC:全称,Inversion of Control,翻译过来叫控制反转。什么意思呢?简单地说就是在不使用IOC时,类A若想获得一个对象,它需要new一个对象,从主从关系上说,这个对象是类A创建的。使用IOC之后,类A若想获得一个对象,它不再自己创建对象,只需要从某个地方获得一个已存在的对象。这个时候创建对象的控制权已经反转,从类A转到了某个地方。使用Spring框架之后这个“地方”和这个“对象”都是Spring提供给我们的。

IOC的好处

最开始遇到IOC的时候,我们都不理解为什么要用IOC,不用不也挺好的吗?其实一点都不好,请看下面叙述。

假设我们有接口:UserDao,类:UserDaoImpl、UserDaoMybatisImpl。在SpringTest中我们需要对数据库进行操作时,创建UserDaoImpl的实例,在最最开始我们连多态都不知道的情况下会使用UserDaoImpl udl = new UserDaoImpl();。这个时候程序的可拓展性是最差的,动一发而牵全身,比如,如果我们把实例更换为UserDaoMybatisImpl,需要修改代码为UserDaoMybatisImpl udl = new UserDaoMybatisImpl();,而且后续使用的方法可能还需要修改。

知道多态后,我们把需求放在接口中,让UserDaoMybatisImpl和UserDaoImpl去实现接口,修改代码的时候只需要从UserDao udl = new UserDaoImpl();修改为UserDao udl = new UserDaoMybatisImpl();,由于需求实在接口中定义的,两个实现类实现需求的规则一致,后序代码几乎不需要怎么修改。详见:接口&多态理解

但是此时UserDao和UserDaoImpl之间以及UserDao和UserDaoMybatisImpl之间存在依赖关系,而且对于程序员也不是很友好,程序员必须得知道UserDao和它的实现类才能进行操作,如果能报获得UserDao实现类这一步交给某个类来做就太好了。所以出现了工厂模式。在工厂模式中,实现类的获取交给工厂来负责,解耦了接口和它的实现类。详见:工厂模式

这时候还存在问题,假如程序员A负责工厂的创建,程序员B负责UserDao类的实现类的开发。B的工作必须先于A完成,否则A得不到UserDao的实现类只能干坐着。为了解决这个问题,反射就派上了用场,由于反射可以根据字符串创建类的实例,所以只要A知道了B负责的类的全称,它不用等到B完成工作就可以干活。

但是字符串写在源文件中不是很好的选择,如上例,我们需要修改的时候还是需要修改源文件,所以配置文件(propertiesxml)就派上了用场。有了配置文件后,我们把字符串写在配置文件中,不需要修改代码就能更换接口实例。

综上:接口 + 工厂模式 + 反射 + 配置文件实现了代码的解耦和,增强了代码的可维护性和可拓展性。这四者也就是Spring的IOC所做的事。

简单的例子

  1. 引入约束:

    <?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">
    
    </beans>
  2. 创建接口:UserDao,类:UserDaoImpl和UserDaoMybatisImpl

    public interface UserDao {
        public void save();
    }
    public class UserDaoImpl implements UserDao{
    
        @Override
        public void save() {
            System.out.println("UserDaoImpl is saving ...");
        }
    }
    public class UserDaoMybatisDao implements UserDao{
    
        @Override
        public void save() {
            System.out.println("UserDaoMybatisDao is saving ...");
        }
    }
  3. 使用Spring调用UserDaoImpl
    配置bean属性:<bean id="UserDao" class="com.spring.firstday.UserDaoImpl"></bean>

    import org.springframework.context.ApplicationContext;
    import org.springframework.context.support.ClassPathXmlApplicationContext;
    
    public class Test {
        public static void main(String[] args) {
            ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
            UserDao ud = (UserDao)ac.getBean("UserDao");
            ud.save();
        }
        /*Console:
         *      UserDaoImpl is saving ...
         * */
    }

    如果想换成UserDaoMybatisImpl,更换一下全类名。<bean id="UserDao" class="com.spring.firstday.UserDaoMybatisDao"></bean>

    import org.springframework.context.ApplicationContext;
    import org.springframework.context.support.ClassPathXmlApplicationContext;
    
    public class Test {
        public static void main(String[] args) {
            ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
            UserDao ud = (UserDao)ac.getBean("UserDao");
            ud.save();
        }
        /*Console:
         *      UserDaoMybatisDao is saving ...
         * */
    }

猜你喜欢

转载自blog.csdn.net/qq_38206090/article/details/82469617