看部电影,透透彻彻理解IoC(你没有理由再迷惑!)

引述:IoC(控制反转:Inverse of Control)是Spring容器的内核,AOP、声明式事务等功能在此基础上开花结果。但是IoC这个重要的概念却比较晦涩隐讳,不容易让人望文生义,这不能不说是一大遗憾。不过IoC确实包括很多内涵,它涉及代码解耦、设计模式、代码优化等问题的考量,我们打算通过一个小例子来说明这个概念。

通过实例理解IoC的概念

贺岁大片在中国已经形成了一个传统,每到年底总有多部贺岁大片纷至沓来让人应接不暇。在所有贺岁大片中,张之亮的《墨攻》算是比较出彩的一部。该片讲述了战国时期墨家人革离帮助梁国反抗赵国侵略的个人英雄主义故事,恢宏壮阔、浑雄凝重的历史场面相当震撼。其中有一个场景:当刘德华所饰演的墨者革离到达梁国都城下,城上梁国守军问到:“来者何人?”刘德华回答:“墨者革离!”我们不妨通过一个Java类为这个“城门叩问”的场景进行编剧,并借此理解IoC的概念:

代码清单3-1  MoAttack:通过演员安排剧本

Java代码  

  1. public class MoAttack {
  2. public void cityGateAsk(){
  3. //①演员直接侵入剧本
  4. LiuDeHua ldh = new LiuDeHua();
  5. ldh.responseAsk("墨者革离!");
  6. }
  7. }

我们会发现以上剧本在①处,作为具体角色饰演者的刘德华直接侵入到剧本中,使剧本和演员直接耦合在一起(图3-1)。 
 
   一个明智的编剧在剧情创作时应围绕故事的角色进行,而不应考虑角色的具体饰演者,这样才可能在剧本投拍时自由地遴选任何适合的演员,而非绑定在刘德华一人身上。通过以上的分析,我们知道需要为该剧本主人公革离定义一个接口:

代码清单3-2  MoAttack:引入剧本角色

Java代码  

  1. public class MoAttack {
  2. public void cityGateAsk()
  3. {
  4. //①引入革离角色接口
  5. GeLi geli = new LiuDeHua();
  6. //②通过接口开展剧情
  7. geli.responseAsk("墨者革离!");
  8. }
  9. }

在①处引入了剧本的角色——革离,剧本的情节通过角色展开,在拍摄时角色由演员饰演,如②处所示。因此墨攻、革离、刘德华三者的类图关系如图 3 2所示: 
 
   可是,从图3 2中,我们可以看出MoAttack同时依赖于GeLi接口和LiuDeHua类,并没有达到我们所期望的剧本仅依赖于角色的目的。但是角色最终必须通过具体的演员才能完成拍摄,如何让LiuDeHua和剧本无关而又能完成GeLi的具体动作呢?当然是在影片投拍时,导演将LiuDeHua安排在GeLi的角色上,导演将剧本、角色、饰演者装配起来(图3-3)。 
 
通过引入导演,使剧本和具体饰演者解耦了。对应到软件中,导演像是一个装配器,安排演员表演具体的角色。 
   现在我们可以反过来讲解IoC的概念了。IoC(Inverse of Control)的字面意思是控制反转,它包括两个内容:

  • 其一是控制
  • 其二是反转

那到底是什么东西的“控制”被“反转”了呢?对应到前面的例子,“控制”是指选择GeLi角色扮演者的控制权;“反转”是指这种控制权从《墨攻》剧本中移除,转交到导演的手中。对于软件来说,即是某一接口具体实现类的选择控制权从调用类中移除,转交给第三方决定。 
   因为IoC确实不够开门见山,因此业界曾进行了广泛的讨论,最终软件界的泰斗级人物Martin Fowler提出了DI(依赖注入:Dependency Injection)的概念用以代替IoC,即让调用类对某一接口实现类的依赖关系由第三方(容器或协作类)注入,以移除调用类对某一接口实现类的依赖。“依赖注入”这个名词显然比“控制反转”直接明了、易于理解。

IoC的类型

从注入方法上看,主要可以划分为三种类型:构造函数注入、属性注入和接口注入。Spring支持构造函数注入和属性注入。下面我们继续使用以上的例子说明这三种注入方法的区别。

构造函数注入

在构造函数注入中,我们通过调用类的构造函数,将接口实现类通过构造函数变量传入,如代码清单3-3所示:

代码清单3-3  MoAttack:通过构造函数注入革离扮演者

Java代码  

  1. public class MoAttack {
  2. private GeLi geli;
  3. //①注入革离的具体扮演者
  4. public MoAttack(GeLi geli){
  5. this.geli = geli;
  6. }
  7. public void cityGateAsk(){
  8. geli.responseAsk("墨者革离!");
  9. }
  10. }

MoAttack的构造函数不关心具体是谁扮演革离这个角色,只要在①处传入的扮演者按剧本要求完成相应的表演即可。角色的具体扮演者由导演来安排,如代码清单3-4所示:

代码清单3-4  Director:通过构造函数注入革离扮演者

Java代码  

  1. public class Director {
  2. public void direct(){
  3. //①指定角色的扮演者
  4. GeLi geli = new LiuDeHua();
  5. //②注入具体扮演者到剧本中
  6. MoAttack moAttack = new MoAttack(geli);
  7. moAttack.cityGateAsk();
  8. }
  9. }

在①处,导演安排刘德华饰演革离的角色,并在②处,将刘德华“注入”到墨攻的剧本中,然后开始“城门叩问”剧情的演出工作。

属性注入

有时,导演会发现,虽然革离是影片《墨攻》的第一主角,但并非每个场景都需要革离的出现,在这种情况下通过构造函数注入相当于《墨攻》剧组每时每刻都要革离的饰演者在场,可见并不符合实际,这时可以考虑使用属性注入。属性注入可以有选择地通过Setter方法完成调用类所需依赖的注入,意为着《墨攻》剧组可在桥段需要时让革离的饰演者在场:

代码清单3-5  MoAttack:通过Setter方法注入革离扮演者

Java代码  

  1. public class MoAttack {
  2. private GeLi geli;
  3. //①属性注入方法
  4. public void setGeli(GeLi geli) {
  5. this.geli = geli;
  6. }
  7. public void cityGateAsk() {
  8. geli.responseAsk("墨者革离");
  9. }
  10. }

MoAttack在①处为geli属性提供一个Setter方法,以便让导演在需要时注入geli的具体扮演者。

代码清单3-6  Director:通过Setter方法注入革离扮演者

Java代码  

  1. public class Director {
  2. public void direct(){
  3. GeLi geli = new LiuDeHua();
  4. MoAttack moAttack = new MoAttack();
  5. //①调用属性Setter方法注入
  6. moAttack.setGeli(geli);
  7. moAttack.cityGateAsk();
  8. }
  9. }

和通过构造函数注入革离扮演者不同,在实例化MoAttack剧本时,并未指定任何扮演者,而是在实例化MoAttack后,在需要革离出场时,才调用其setGeli()方法注入扮演者。按照类似的方式,我们还可以分别为剧本中其他诸如梁王、巷淹中等角色提供注入的Setter方法,这样,导演就可以根据所拍剧段的不同,注入相应的角色了。

接口注入

将调用类所有依赖注入的方法抽取到一个接口中,调用类通过实现该接口提供相应的注入方法。为了采取接口注入的方式,必须先声明一个ActorArrangable接口:

Java代码  

  1. public interface ActorArrangable {
  2. void injectGeli(GeLi geli);
  3. }

然后,MoAttack实现ActorArrangable接口提供具体的实现:

代码清单3-7  MoAttack:通过接口方法注入革离扮演者

Java代码

  1. public class MoAttack implements ActorArrangable {
  2. private GeLi geli;
  3. //①实现接口方法
  4. public void injectGeli (GeLi geli) {
  5. this.geli = geli;
  6. }
  7. public void cityGateAsk() {
  8. geli.responseAsk("墨者革离");
  9. }
  10. }

Director通过ActorArrangable的injectGeli()方法完成扮演者的注入工作。

代码清单3-8  Director:通过接口方法注入革离扮演者

Java代码  

  1. public class Director {
  2. public void direct(){
  3. GeLi geli = new LiuDeHua();
  4. MoAttack moAttack = new MoAttack();
  5. moAttack. injectGeli (geli);
  6. moAttack.cityGateAsk();
  7. }
  8. }

由于通过接口注入需要额外声明一个接口,增加了类的数目,而且它的效果和属性注入并无本质区别,因此我们不提倡采用这种方式。

通过容器完成依赖关系的注入

虽然MoAttack和LiuDeHua实现了解耦,MoAttack无须关注角色实现类的实例化工作,但这些工作在代码中依然存在,只是转移到Director类中而已。假设某一制片人想改变这一局面,在选择某个剧本后,希望通过一个“海选”或者第三中介机构来选择导演、演员,让他们各司其职,那剧本、导演、演员就都实现解耦了。 
   所谓媒体“海选”和第三方中介机构在程序领域即是一个第三方的容器,它帮助完成类的初始化与装配工作,让开发者从这些底层实现类的实例化、依赖关系装配等工作中脱离出来,专注于更有意义的业务逻辑开发工作。这无疑是一件令人向往的事情,Spring就是这样的一个容器,它通过配置文件或注解描述类和类之间的依赖关系,自动完成类的初始化和依赖注入的工作。下面是Spring配置文件的对以上实例进行配置的配置文件片断:

Xml代码  

  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:p="http://www.springframework.org/schema/p"
  5. xsi:schemaLocation="http://www.springframework.org/schema/beans
  6. http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
  7. <!--①实现类实例化-->
  8. <bean id="geli" class="LiuDeHua"/>
  9. <bean id="moAttack" class="com.baobaotao.ioc.MoAttack"
  10. p:geli-ref="geli"/><!--②通过geli-ref建立依赖关系-->
  11. </beans>

通过new XmlBeanFactory(“beans.xml”)等方式即可启动容器。在容器启动时,Spring根据配置文件的描述信息,自动实例化Bean并完成依赖关系的装配,从容器中即可返回准备就绪的Bean实例,后续可直接使用之。 
   Spring为什么会有这种“神奇”的力量,仅凭一个简单的配置文件,就能魔法般地实例化并装配好程序所用的Bean呢?这种“神奇”的力量归功于Java语言本身的类反射功能。

转载请注明出处:http://www.cnblogs.com/ji-yun/p/5804591.html

时间: 2024-12-14 10:25:44

看部电影,透透彻彻理解IoC(你没有理由再迷惑!)的相关文章

透透彻彻IoC

本文转载自:http://stamen.iteye.com/blog/1489223/ 引述:IoC(控制反转:Inverse of Control)是Spring容器的内核,AOP.声明式事务等功能在此基础上开花结果.但是IoC这个重要的概念却比较晦涩隐讳,不容易让人望文生义,这不能不说是一大遗憾.不过IoC确实包括很多内涵,它涉及代码解耦.设计模式.代码优化等问题的考量,我们打算通过一个小例子来说明这个概念. 通过实例理解IoC的概念 贺岁大片在中国已经形成了一个传统,每到年底总有多部贺岁大

透透彻彻IoC(你没有理由不懂!)

http://www.myexception.cn/open-source/418322.html 引述:IoC(控制反转:Inverse of Control)是Spring容器的内核,AOP.声明式事务等功能在此基础上开花结果.但是IoC这个重要的概念却比较晦涩隐讳,不容易让人望文生义,这不能不说是一大遗憾.不过IoC确实包括很多内涵,它涉及代码解耦.设计模式.代码优化等问题的考量,我们打算通过一个小例子来说明这个概念. 通过实例理解IoC的概念 贺岁大片在中国已经形成了一个传统,每到年底总

作为一个程序员,没有看过这几部电影你就out了!

摘要:每次看科技类的电影,都感觉很酷,一直以为那是虚幻的,实际上是现实中的一些技术在电影中被无限扩大了,而并不是不存在的.其实很多时候在不同的行业领域中,都分享着一个共同的理念和思想,就像很多编程中涉及到的思想被运用到了电影里. 艺术源于生活,高于生活.当计算机编程中的某些概念被运用到电影里之后,我们总会觉得电影很酷,很不可思议,其实是那些编程思想被艺术化了.既然说艺术源于生活并高于生活,反过来这些电影对程序员更好的理解这些编程思想也会有很好的帮助.以下小编为大家推荐几部非常值得一看的电影. 1

100部电影,100种人生,你看过几部?

100部电影,100种人生,你看过几部? 王小欢 ▲ 1.居家男人 ▲ 2.当幸福来敲门 ▲ 3.少林足球 ▲ 4.这个杀手不太冷 ▲ 5.暴力街区 ▲ 6.超脱 ▲ 7.大话西游 ▲ 8.第11小时 ▲ 9.观音山 ▲ 10.国王的演讲 ▲ 11.秋之白华 ▲ 12.熔炉 ▲ 13.少年斯派维的奇异旅行 ▲ 14.她 ▲ 15.新桥恋人 ▲ 16.一代宗师 ▲ 17.one day ▲ 18.V字仇杀队 ▲ 19.爱你 ▲ 20.爱情与灵药 ▲ 21.傲慢与偏见 ▲ 22.巴黎小情歌 ▲ 23

人一生必看的100部电影(全球最佳电影排名榜TOP250)

人一生必看的100部电影(全球最佳电影排名榜TOP250) 人的一生能看多少部电影?假设我们每周都看一部,从10岁看到80岁将会看3640部.但是我们也不可能喜欢这全部的电影.大多数的可能,我们会根据朋友,或者其他来源的推荐去观看.如果看的影片不合自己口味,我们会很悲哀.本次的榜单是亚马逊公司旗下的IMDB数据库筛选出的全球最佳电影排行榜.这里有全球无数人同样喜爱的各类电影,从中获取你还没看过的电影将是最好的选择! 第001名 教父 The Godfathertop (1972) 第002名 肖

透彻理解Ioc

引述:IoC(控制反转:Inverse of Control)是Spring容器的内核,AOP.声明式事务等功能在此基础上开花结果.但是IoC这个重要的概念却比较晦涩隐讳,不容易让人望文生义,这不能不说是一大遗憾.不过IoC确实包括很多内涵,它涉及代码解耦.设计模式.代码优化等问题的考量,我们打算通过一个小例子来说明这个概念. 通过实例理解IoC的概念 贺岁大片在中国已经形成了一个传统,每到年底总有多部贺岁大片纷至沓来让人应接不暇.在所有贺岁大片中,张之亮的<墨攻>算是比较出彩的一部.该片讲述

大学里的十件事、十本书、十部电影

:[大学里的十件事.十本书.十部电影!(精华版)]1.大学里要做的10件事1.学习计算机,通过二级考试,并学习其它电脑知识.不是为了拿到证书而炫耀,而是为了以后学习其它电脑知识的便利2.认真学习英语,顺利通过四级考试,并准备六级.不仅是为了自己以后的毕业证,更是为了多一种在必要的时候表达自己的方式和能力3.有时间写作,写一段自传.不是为了出书,而是要让自己记住曾经有过一段经历,不要忘记自己的生活4.有条件的话恋爱一次5.要努力学习.不是为了通过考试,那是太容易的事.不论是否满意,既然已经进了大学

深入理解IOC模式及Unity简单应用

研究了下,有几篇博客确实已经说得很清楚了 1.IoC模式:http://www.cnblogs.com/qqlin/archive/2012/10/09/2707075.html  这篇博客是通过一个播放器的例子来说明什么是依赖,依赖倒置,控制反转(IOC),最后实现依赖注入.通过Unity实现IOC容器.不错的一个例子 2.深入理解DIP.IoC.DI以及IoC容器 这个算是最通俗易懂的,手动实现了IOC容器  由浅入深 3.理解依赖注入(IOC)和学习Unity 这个也不错,特别最后介绍的挺

值得反复体会的几部电影

我是一个怀旧的人,喜欢听老音乐,看老电影.下面的几部电影,我每隔一段时间都会看一遍.     1<肖申克的救赎> <肖申克的救赎>,是我上大一的时候,英语老师推荐的电影.刚看第一遍的时候,就被主人公安迪毅力,向往自由的精神所鼓舞.下面摘录了一段观后感,我觉得说的很好,值得我们去深思! 在我们自己的现实中,我们注定是自己剧中的主角,而电影给我们一个局外人的机会,所以我们经常喜欢以一个局外人角色从影片中寻找自己.审视自己,总喜欢从电影中寻找生活.审视现实.这一次我看到最多的是体制下的自