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完成工作就可以干活。
但是字符串写在源文件中不是很好的选择,如上例,我们需要修改的时候还是需要修改源文件,所以配置文件(properties和xml)就派上了用场。有了配置文件后,我们把字符串写在配置文件中,不需要修改代码就能更换接口实例。
综上:接口 + 工厂模式 + 反射 + 配置文件实现了代码的解耦和,增强了代码的可维护性和可拓展性。这四者也就是Spring的IOC所做的事。
简单的例子
引入约束:
<?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>
创建接口: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 ..."); } }
使用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 ... * */ }