spring学习笔记-DI

spring开源框架可以说对于java生态起到了一个基石的作用,而spring的两大核心功能IOC容器和AOP可以说支撑了spring框架的流行,这篇文章主要介绍一下ioc容器的一些核心观点。

一、什么依赖注入:

ioc的英文全称是(Inversion of controll),可以翻译成控制反转,其实就是一种权利的反转,那到底是掌握了什么样的权利呢?我们先了解一下DI(依赖注入)我们通过下面的几个例子来深入的讨论一下。

  • 生活小场景:
    我们需要发送一封电子邮件,我们一般需要几个步骤:
    1、需要一个邮件的编辑器;
    2、需要一个语法拼写检查的工具和翻译工具;
    3、需要一个客户端发送;
    对于万物皆对象的java语言来说,我们好像很难将这一系列操作封装成一个统一的操作,我们试者来分解一下。
客户端:Emailer
public class Emailer{
    public void send(String text){
     //发送邮件
    }
}
但是我们需要在发送邮件以前,进行一个翻译和语法检查,比如我们的客户为美国人,so
翻译并语法检查的SpellChecker
public class SpellChecker{

     public String checkAndTranslation(String text){
         //翻译并且语法检查。
     }
}

现在我们在完成邮件发送整个流程的时候,需要两个对象的关联操作,所以在java语言中我们
可能需要两个实例,所以我们可以说Emailer和SpellChecker是有依赖关系的。万物皆有关联
同样在代码中也体现的淋漓精致。
我们来研究一下这两个对象的依赖关系,首先Emailer必须需要SpellChecker才可以完成发送邮
件的功能,并且SpellChecker必须在Emailer中才能发挥自己的功能,看一下一个总结语句:
Dependency—A specific service that is required by another object to fulfill its function.
Dependent—A client object that needs a dependency (or dependencies) in order to perform
its function.
我相信已经可以很深刻的理解上述的两个总结语句了。

那我们需要将这两个对象依赖起来可以怎么做呢:

  • 手动构造方法进行依赖关联:
public class Emailer{
  private SpellChecker spellChecker;
  public Emailer(){
     this.spellChecker=new SpellChecker();
  }
    public void send(String text){
     //发送邮件
    }
}
1、在构造方法中新建spellChecker实例,在Emailer初始化实例的时候,自然包含了spellChecker实例。
  • 思考一下这种方式的一个弊端:
    依赖关系仿佛成了一种寄生关系,两个对象紧紧的耦合在一起,可以说SpellChecker是Emailer内部的一部分了,无法测试无法扩展,甚至当客户变为日本人和法国人的时候,我们甚至一时间不能支持。
  • 构造器传入参数的方式
public class Emailer {
private SpellChecker spellChecker;
public void setSpellChecker(SpellChecker spellChecker) {
this.spellChecker = spellChecker; }
}

//Emailer emailer=new Emailer(new SpellChecker());

注意我对代码进行了一点修改,构造器中接受了一个SpellChecker,我们可以在创建Emailer的同时,决定传入什么模式的拼写检查器。好像我们平时大游戏的时候根据boss不同更换不同的装备一样,好像一种把一个对象注入另一个对象之中,我们称之为依赖的注入,也就是经常听见的DI,而这种通过构造方法注入的形式我们称之为constructor injection。

  • setter方法进行注入
    其实和构造方法异曲同工,通过setter方法将一个对象注入另一个对象,这里就不使用代码演示了。但是这个两种方法有没有什么弊端,我们必须要知道依赖的一个关系,同样我们如果在很多地方都需要依赖于一个相同的对象,我们可能需要很多的重复代码,甚至是一个小的依赖改动,我们也需要很多的代码修改,思考如何才能减轻依赖之间的负担呢,有没有一种生产工具,我们需要什么给我们什么。比如我们需要一个发送邮件的东西,就给我们一个发送邮件的东西,我们只需要调用这个东西,而不需要关心里面的构造如何。
  • 工厂模式:核心思想就是把这些复杂的依赖交给一个第三方去做,我们称之为工厂,这个很贴近生活,我们只需要把想法告诉工厂,然后工厂就给我们生产出想要的商品,我们不需要关心这个商品制作中需要的种种原料,就是这个意思。
public class EmailerFactory {
            public Emailer newFrenchEmailer() {
                return new Emailer(new FrenchSpellChecker());
            }
}
上述代码就是委托工厂给我们一个Emailer,包含了一个法语的翻译拼写检查器。我们会发现现在这两个对象,不管是Emailer还是SpellChecker的关系变得干净了很多,没有了谁包含谁的现象,都由工厂帮我们搞定了。

Emailer emailer=new EmailerFactory().newFrenchEmailer()
这样的代码是不是很清晰,我们只需要知道我们需要什么样的东西而选择什么样的工厂。
看起来工厂模式已经很好的解决了手动注入的问题,但是我们需要思考一下有什么弊端吗?在工厂中如果我们维护了很多的依赖,依赖中又依赖的情况,我们在添加一种方法的时候还是需要编写一堆的代码。

  • The Service Locator pattern(服务定位模式)
    其实这种模式也是一种特殊的工厂模式,下面的代码就是一种服务定位模式:
    Emailer emailer = (Emailer) new ServiceLocator().get("Emailer");
    可以使用kv的形式,用k拿到对应的一个v,而v就是我们需要的服务;java中常见的服务定位模式JNDI(java name and Directory interface)我们也可以理解为发布一个服务,我们根据需要去订阅,发布的服务的命名标准可以就是一个key。但是这种方式太过依赖于这个key,key的书写错误,会导致错误的对象转化错误。

二、DI vs IOC

"Don’t call us; we’ll call you!" 就像好莱坞挑选演员一样。这种来帮助我们管理这些对象之间的依赖蓝图的我们就称之为依赖注入框架,或者依赖注入器,而spring的IOC容器就是一个依赖注入的框架,也就是说IOC容器和DI可以画上等号。什么样的东西可以称之为使用了DI或者是一个IOC容器。下面我们已经更能理解一下几个术语:
1、Dependency injector:依赖注入器,其实就是一个框架或者一个类库提供依赖注入的方法;
2、Dependency injection:依赖注入就是上述讲述的一种处理依赖关系的方案,也可以说是一种模式。

  • spring的IOC就是DI的一个很好的实现框架,也就是说是一个优秀的依赖注入器,其中spring更是将注解,反射,泛型,代理这些技术发挥的更好;但是不是说只有spring的ioc容器使用了DI,其实还有很多框架都是DI的完美实现,这个有兴趣可以查看一下。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,816评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,729评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,300评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,780评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,890评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,084评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,151评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,912评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,355评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,666评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,809评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,504评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,150评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,121评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,628评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,724评论 2 351