一、什么是 IOC


IoC就是Inversion of Control,控制反转。在Java开发中,IoC意味着将你设计好的类交给系统去控制,而不是在你的类内部控制。这称为控制反转。


 


下面我们以几个例子来说明什么是IoC


 


假设我们要设计一个Girl和一个Boy类,其中Girl有kiss方法,即Girl想要Kiss一个Boy。那么,我们的问题是,Girl如何能够认识这个Boy?



    在我们中国,常见的MM与GG的认识方式有以下几种


    1 青梅竹马; 2 亲友介绍; 3 父母包办


 


    那么哪一种才是最好呢?


    青梅竹马:Girl从小就知道自己的Boy。


   



   


 


publicclass        Girl {  
                  void        kiss(){
              Boy boy =        new        Boy();
           }
       }



 


 


    然而从开始就创建的Boy缺点就是无法在更换。并且要负责Boy的整个生命周期。如果我们的Girl想要换一个怎么办?(笔者严重不支持Girl经常更换Boy)


 


    亲友介绍:由中间人负责提供Boy来见面



       


publicclass        Girl {
                  void        kiss(){
              Boy boy = BoyFactory.createBoy();      
           }
       }


 


    亲友介绍,固然是好。如果不满意,尽管另外换一个好了。但是,亲友BoyFactory经常是以Singleton的形式出现,不然就是,存在于Globals,无处不在,无处不能。实在是太繁琐了一点,不够灵活。我为什么一定要这个亲友掺和进来呢?为什么一定要付给她介绍费呢?万一最好的朋友爱上了我的男朋友呢?


 


    父母包办:一切交给父母,自己不用费吹灰之力,只需要等着Kiss就好了。


 



      



public        
class Girl {
    void kiss(Boy boy){
       // kiss boy  
      boy.kiss();
    }
}


    Well,这是对Girl最好的方法,只要想办法贿赂了Girl的父母,并把Boy交给他。那么我们就可以轻松的和Girl来Kiss了。看来几千年传统的父母之命还真是有用哦。至少Boy和Girl不用自己瞎忙乎了。


    这就是IOC,将对象的创建和获取提取到外部。由外部容器提供需要的组件。


 


    我们知道 好莱坞原则“Do not call us, we will call you.” 意思就是,You, girlie, do not call the boy. We will feed you a boy。


 


    我们还应该知道 依赖倒转原则


 



Eric Gamma说,要面向抽象编程。面向接口编程是面向对象的核心。


组件应该分为两部分,即


Service, 所提供功能的声明


Implementation, Service的实现


好处是:多实现可以任意切换,防止 “everything depends on everything” 问题.即具体依赖于具体。


所以,我们的Boy应该是实现Kissable接口。这样一旦Girl不想kiss可恶的Boy的话,还可以kiss可爱的kitten和慈祥的grandmother。


 



二、 IOC type


    IoC的Type指的是Girl得到Boy的几种不同方式。我们逐一来说明。


 


    IOC type 0:不用IOC


 


 



publicclass        Girl        implements        Servicable {


           private        Kissable kissable;


           public        Girl() {
               kissable =        new        Boy();
           }

           publicvoid        kissYourKissable() {
               kissable.kiss();
           }

}



 


 


    Girl自己建立自己的Boy,很难更换,很难共享给别人,只能单独使用,并负责完全的生命周期。


 


    IOC type 1 先看代码:


 


 


publicclass        Girl        implements        Servicable {


    Kissable kissable;


           publicvoid        service(ServiceManager mgr) {
               kissable = (Kissable) mgr.lookup(       “kissable”);
           }

           publicvoid        kissYourKissable() {
               kissable.kiss();
           }

}



 


 


    这种情况出现于Avalon Framework。一个组件实现了Servicable接口,就必须实现service方法,并传入一个ServiceManager。其中会含有需要的其它组件。只需要在service方法中初始化需要的Boy。


    另外,J2EE中从Context取得对象也属于type 1。


 


    它依赖于配置文件


 



<container>
           <component name=       “       kissable       “        class=       “       Boy">              
       <configuration>        …        </configuration>
           </component>

           <component name=“girl" class=“Girl" />       </container>



 


 


    IOC type 2:


   


 


publicclass        Girl {


           private        Kissable kissable;


           publicvoid        setKissable(Kissable kissable) {
                      this       .kissable = kissable;
           }

           publicvoid        kissYourKissable() {
               kissable.kiss();
           }

}



 


 


    Type 2出现于Spring Framework,是通过JavaBean的set方法来将需要的Boy传递给Girl。它必须依赖于配置文件。


       


 


<beans>
           <bean id=       “       boy" class=       “       Boy"/>
           <bean id=       “       girl       “        class=       “       Girl">
               <property name=“kissable">           <ref bean=“boy"/>               </property>
           </bean>
       </beans>



 


 


IOC type 3


 


 



publicclass        Girl {


           private        Kissable kissable;


           public        Girl(Kissable kissable) {
                      this       .kissable = kissable;
           }

           publicvoid        kissYourKissable() {
               kissable.kiss();
           }

}



 


 


    这就是PicoContainer的组件 。通过构造函数传递Boy给Girl。


 


 


 



PicoContainer container =        new        DefaultPicoContainer();
       container.registerComponentImplementation(Boy.       class       );
       container.registerComponentImplementation(Girl.       class       );
       Girl girl = (Girl) container.getComponentInstance(Girl.       class       );
       girl.kissYourKissable();



三、IOC的type

3.1 IoC模式简介

    IoC(Inversion of Control)模式并不是什么新的东西,它是一种很普遍的概念,GoF中的Template Method 就是IoC的结构。顾名思义,IoC即控制反转。著名的好莱坞原则:“Don’t Call us, We will call you”,以及Robert C. Martin在其敏捷软件开发中所描述的依赖倒置原则(Dependency Inversion Principle, DIP)都是这一思想的体现。依赖注入(Dependency Injection)是Martin Flower对IoC模式的一种扩展的解释[2]。IoC是一种用来解决组件(实际上也可以是简单的Java类)之间依赖关系、配置及生命周期的设计模式,其中对组件依赖关系的处理是IoC的精华部分。IoC的实际意义就是把组件之间的依赖关系提取(反转)出来,由容器来具体配置。这样,各个组件之间就不存在hard-code的关联,任何组件都可以最大程度的得到重用。运用了IoC模式后我们不再需要自己管理组件之间的依赖关系,只需要声明由容器去实现这种依赖关系。就好像把对组件之间依赖关系的控制进行了倒置,不再由组件自己来建立这种依赖关系而交给容器(例如我们后面会介绍的PicoContainer、Spring)去管理。

    我们从一个简单的例子看起,考虑一个Button控制Lamp的例子:

public class Button { 
  
    private Lamp lamp; 
  
    public void push() { 
  
        lamp.turnOn(); 
  
    } 
  
}

    但是马上发现这个设计的问题,Button类直接依赖于Lamp类,这个依赖关系意味着当Lamp类修改时,Button类会受到影响。此外,想重用Button类来控制类似与Lamp的(比如同样具有turnOn功能的Computer)另外一个对象则是不可能的。即Button控制Lamp,并且只能控制Lamp。显然违反了“高层模块不应该依赖于低层模块,两者都应该依赖于抽象;抽象不应该依赖于具体实现,细节应该依赖于抽象” 这一原则(DIP原则)。考虑到上述问题,自然的想到应该抽象出一个接口SwitchableDevice,来消除Button对Lamp的依赖,于是设计如下:

public class Button { 
  
    private SwitchableDevice lamp; 
  
    public Button(){ 
  
    lamp= new Lamp(); 
  
    } 
  
}

    再深入考虑一下,虽然我们的Button现在可以控制实现了SwitchableDevice接口的Computer,但是Button和Lamp类之间还是存在create这样的依赖关系。为了解决这种依赖关心,经典的GoF模式就是采用Factory模式,将对象的创建交给Factory类来创建,但是这种创建仍是显示的,组件变化了仍然需要重新编译程序。而采用J2EE经典的service locator模式,如果你要把Button组件拿到另一个系统里面用,你就必须修改它的源码,让它使用另一个系统的serviceLocator。换句话说,这个组件不具备可移植性。这就是需要依赖注入的道理,让组件的创建、配置及生命周期总是由外部容器来管理。

 

3.2 IoC的类型

3.2.1 IoC的类型

    在介绍如何利用IoC模式实现彻底解耦之前,我们先看看IoC的类型:

3.2.1.1 Method-based (M) IoC

    在每个方法调用中传递其依赖的组件。如果方法需要某个组件,就把该组件作为参数传递给方法。

3.2.1.2 Interface-based (I) IoC (Type 1)

    使用接口如Serviceable, Configurable 等等,来声明依赖。EJB容器就是一个Type1的重量级容器,部署在它内部的EJB组件使用接口来声明依赖关系。

3.2.1.3 Setter-based (S) IoC (Type 2)

    使用setters 来设置依赖组件。把依赖的组件作为一个属性,通过setters方法来动态设置依赖组件。

3.2.1.4 Constructor-based (C) IoC (Type 3)

    使用构造函数来声明依赖。通过传递组件参数到构造函数中,来实现依赖关系。

 

3.2.2 IoC类型的比较

    这几种类型中,type 3侵入性较小。因为在面向对象的理论里,constructor并不是对象契约的一部分。按照Bertrand Meyer的说法,你永远不应该直接调用constructor,因为这就意味着client代码与实现(而非契约)绑定在一起。那么,既然constructor并不属于对象契约的一部分,在constructor里暴露元信息就不会影响对象契约。Type 2虽然也很好,但setter毕竟属于对象契约,把一个setter用于IoC多少有一点“破坏性”,而且通过setter方法过多的暴露了内部对象的内部细节,这就失去了对象的封装。

    Type 2很合适的作为应用程序的bean工厂。如果是更多的动态组装,可能type 3更好一点。从定义上来说,type 2是基于setter的,type 3是基于constructor的。为什么说type 2更适合于做bean工厂呢?因为setter是各个分离的,对于有定义的n个setter,bean工厂调用其中的0~n个都是合法的。而type 3则稍微有点麻烦,不能适应依赖较多的情况,组件的“元信息”在constructor的参数列表中体现,你必须一次性提供所有必要的参数。如果需要很多组件,就需要在构造函数中传递很多参数,这样会导致constructor的参数过多过长。

 

3.3 IoC容器

    根据容器对组件的侵入的程度,可以把IoC容器分为以下三类:

3.3.1 Interface Injection

    对应Type 1 IoC ,使用接口来声明依赖。这类IoC容器侵入性最强,需要通过上下文来获取组件.组件需要实现容器提供的特定接口,这样,组件的重用就被限定在该容器内。这类容器的代表有Apache Avalon。Avalon 不怎么流行,尽管它很强大而且有很长的历史。Avalon属于重量级容器,并且看起来比新的IoC解决方案更具侵入性。

 

3.3.2 Setter Injection

    对应Type 2 IoC ,使用setters来设置依赖组件。这类IoC容器需要组件提供accessor方法,依赖关系通过setter方法来注入。按照java组件模型,一般的javabean都会有accessor方法,因此组件的重用性没有任何限制。这类容器的代表有Spring,同时它也实现了第三类IoC容器。Spring是一个非常活跃的、优秀的开源项目。它是一个基于IoC和AOP(Aspect-Oriented Programming,面向方面编程)的构架多层J2EE系统的框架,它优雅的实现了MVC框架,支持使用可声明事务管理(declarative transaction management)。更重要的是Spring框架的无侵入性[3]

 

3.3.3 Constructor Injection

    对应Type 3 IoC ,使用构造函数来声明依赖。这类IoC容器需要组件由构造方法来配置依赖关系。和第二种IoC类型类似,组件重用没有任何问题。并且Constructor Injection更加严格,完全按照契约(contract)来配置组件依赖。这类容器的代表有PicoContainer。

PicoContainer是一个轻量级而且更强调通过构造函数表达依赖性,而不是JavaBean 属性。 与Spring不同,它的设计允许每个类型一个对象的定义(可能是因为它拒绝任何Java代码外的元数据导致的局限性)。

 

3.4 利用IoC容器实现控制反转

    下面我们就来看看如何利用IoC容器PicoContainer实现本文开始处举的例子,主要代码如下:

private MutablePicoContainer configureContainer() { 
  
    MutablePicoContainer pico = new DefaultPicoContainer(); 
  
    pico.registerComponentImplementation(SwitchableDevice.class, Lamp.class); 
  
    pico.registerComponentImplementation(Button.class); 
  
    return pico; 
  
}

    然后就可以通过MutablePicoContainer的getComponentImplementation方法获得实现类,调用其push方法控制Lamp的开关,这样一来,两者之间的耦合通过PicoContainer提供的Assembler完全消除了。

    Spring则通过一个XML格式的配置文件,将两者联系起来。使用时,通过ApplicationContext获得Button bean,再调用其方法实现,同样也消除了耦合关系。

 

四、总结

    IoC具有以下几个优点:

    1.因为组件不需要在运行时寻找合作者,所以他们可以更简单的编写和维护。由于同             样原因,便于编写测试代码,使类的测试更容易。

    2.不需要外部依赖。能在任何环境下开发和测试组件,而不需要特殊的部署环境,像   JNDI、EJB那样。并且在不同IoC容器中可方便的重用和改变。

    3.整个系统更容易组装和配置。大部分业务对象不依赖于IoC容器的APIs。这使得很  容易使用遗留下来的代码,且很容易的使用对象,无论在容器内或不在容器内。

    4.增加组件的复用程度,提供软件生成效率。

    当然,IoC与通常的方法相比,代码不便于理解,因为组件创建是隐含的。所以轻量级的、无侵入性的IoC容器仍然有待我们去研究开发。