更新下:其实我觉得依赖倒置原则是思路,是系统构建的大方向和哲学,最接近这个哲学的工具是First Class Module和Existance Type。而很多主牛语言根本没这种东西。那么一个比较通用的工具就是DI.
可能文章写得太长了。。。 其实简化来说就两点:
建议把2个逻辑分开:
做到这点需要component不能把自己的依赖写死(不能自己去创建自己需要的dependency的实例),而是只是声明自己需要的spec(一般是interface;如果只存这个spec/interface的一种实例,那么懒得定义interface用实例的类型做依赖声明也可以,需要存在不同实例的时候再重构抽出interface就行了,但是别自己create自己的依赖。比如sort的实现把要sort的compareTo的实例固定,那么sort就只能sort一种T)这就是广义的DI。 无法把广义DI和DI framework分开思考的人就别评论了。谢。
只有很少几种情况components可以直接决定自己的依赖,比如这个依赖绝对不会更改(这意味着你想连UT都把这个components和依赖必须一起测,因为写死了),或者这个实例在世界上只存在一种(比如你依赖于一些数学函数,Math.abs) 这些东西只存在一种实现,绝对不会变,即使放在UT里。
注意:如果需要component控制dependency生成的时机timing,可以inject进来dependency的factory。那么component虽然控制dependency instance生成的时机,但是不知道生成的具体实例类型,同样松耦合了各个Components
再次注意:我个人不喜欢DI Framework,是手动DI,或者用Reader Monad一派的,友军请别乱开枪。
看了很多答案,感觉都没提到依赖注入对系统设计的重要性,所以其实我更想说一下依赖倒置原则和系统设计…
不过先答正题:Go当然也可以做依赖注入:"Dependency Injection" in Golang
因为依赖注入也可以手动完成,而根本不需要DI framework。所以任何支持subtyping或者first class function的语言都完全可以做dependency injection;甚至在C里也可以用函数指针做DI。
有位答主问:
我更好奇的是,这世上有一离开“依赖注入”就玩不转的项目吗?
我想就这个问题讲一下依赖注入,依赖倒置,和系统设计。
依赖注入可以让让“提供功能类A”和“使用功能类B”解耦合;A注入在B里,那么如果A有subtyping的变换A_x,A_y...时,而整体系统调整(比如新feature,测试环境和prod环境)需要B在不同的场景/时间注入其他具体功能实现类A_x...时; B都不需要改,且系统可以通过不同的配置轻松实现在不同场景的多版本系统。最常见的就是业务类使用Dao类,不要在业务类里new具体Dao类,这样就绑死了关系,而注入Dao类,这样则任何Dao的subtyping,业务类都可以接受。
其实依赖注入最重要的作用之一是实现依赖倒置;(控制反转和依赖倒置不同,最后说)
<划重点>让任何B调A的关系,不都产生B必须绑死依赖于A的效果</划重点>。
当B必须用到A的功能的时候,控制流是从B到A,A不需要知道B(最直观的proof A不需要import B),而B必须知道A,或者说B直接依赖于A(直观proof就是B需要import A才能用A),依赖关系箭头从B到A;这样会导致类依赖图甚至包依赖图(如果A和B在不同包)是从B指向(依赖)A。
但是如果有一个interface类C在B的包里(甚至自已的一个interface包里),让A实现C(注意这样A import C依赖于C),让B使用C的interface(B也依赖C)来间接使用C的implementation A,A的实例通过依赖注入到B里供B使用;这样依赖关系就变成了A依赖C,而如果C在B的包里(为什么C放在B的包里合理请往下看),依赖关系就变成了A所在的包依赖于B所在的包(注意本来是B包依赖A包)。从而实现了依赖的倒置。
为什么要这么麻烦做依赖倒置?如果你看过我写的这篇关于系统分层的文章:用谁都能看懂的方式来解释系统设计中的分层,那么应该明白一个稳定易拓展的系统应该让底层依赖高层,而不是相反(高层依赖底层),一个合理的系统,应该由高层按照自己的需求,指定底层应该满足的specification(比如B包是一个高层业务包,那么B里的interface C就是B对于底层的命令式的Specification,然后B里的所有逻辑都应该基于这个specification来写,因为B和B的Specification, 也就是interface C都是在高层抽象讨论问题,所以B和Interface C放在一个高层包里是一种合理设计),底层应该想方设法来满足这个specification(也就是底层类A需要实现interface C);而不应该让业务类扭曲自己来迎合底层类的需要(非依赖倒置的情况);最终实现整个系统的高层底层配合;
用DDD的话来说就是,业务决定技术实现, 而不是相反。
不使用依赖反转的系统构架,控制流和依赖关系流的依赖箭头是一个方向的,由高层指向底层,也就是高层依赖底层,最明显的特点就是高层包需要import很多底层包里的类,这样的话任何的底层小改动,都可能产生影响高层业务逻辑类的改动的蝴蝶效应;这样的系统耦合严重,维护,模拟,测试,实验和拓展都很困难(想想你一个业务类绑死了数据库类,你怎么做UT?怎么在Continuous Delivery的pipeline里模拟数据库做integration test?怎么能简单的把业务数据写替换成写到SQS里,换成写到S3里,写到Kinesis里? 或者能同时都写),可能动不动就要重构和re-architecture;让程序员苦不堪言。而使用依赖倒置,使得底层包依赖高层包。高层包里不会import任何一个底层类。只要interface设计的好,底层的任何变动,都不会影响高层一行code。
严重注意⚠️:一个系统绝不仅仅只有高层和底层2层,而是可以有很多层,业务类也可以并需要分为多层,同时保证依赖倒置!
顺便提一下控制反转,比较容易和依赖倒置原则弄混(太早学的这些,我在开始写这个答案的时候也弄混了这些名词。。。),控制反转是EJB,Spring这种框架平台兴起的时候提起的一个概念。相对于Lib来说,我们总是自己application的逻辑调用Lib的逻辑,控制流由我们的application规定,EJB,Spring是让Framework调用我们的逻辑。在这个控制流的方向上,体现了控制反转。这个原则也称为Hollywood Principle - "Don't call us, we'll call you"。
其实Template Method这个设计模式也体现了控制反转,父类实现一些框架方法并定义控制流如何走,控制流会按照定义好的顺序调用一些abstract方法,然后留给子类去实现这些abstract方法。JUnit框架就是一个好例子。
控制反转是当系统的很多东西都能够模版化的时候,而业务逻辑只是其中的某一步或者几步,那么framework一般需要DI的方式来管理你的领域(业务)类,来inject到自己的控制流里,成为framework规定的控制流的一部分,而你的业务类则不需要关心控制流。可以看出,当控制流可以framework化和模版化的时候,控制反转相当重要(EJB,Spring,JUnit都是例子)。
Dependency Framework, 就是Spring,GoogleGuice,MacWire(一个scala DI framework), 这种通过config file或者annotation等标记来帮助你自动DI的框架,那么到底是手动wire手动DI好,还是用DI framework好呢?
要评价这个就不免带个人意见色彩了。。。 我个人不是很喜欢DI framework,因为我喜欢函数风格的程序,DI对于函数式来说不过是high order function罢了。我个人更倾向于MacWire的作者的意见, In today’s post-OO world, is dependency injection still relevant? 就是DI以后可能会设计为语言自己的功能,由语言来提供一个隐式“environmental” parameter。
Java,Scala里的DI framework有两个好处:
依赖注入,可说是任何程序员必掌握的技巧之一,系统设计入门基础技能之一。而依赖倒置原则作为一条原则,自从被人发现以来,从来都没被颠覆过,可以算得上软件设计艺术的真理之一。
学会一门语言的语法,并不代表你“真的会写程序”了。正如你学会了铅笔的构造原理,你也没法像画家一样画画。真正的画家用任何画笔都能画画,只是有些画笔用着更顺手罢了。
(最后推荐Clean Architecture这本书,除了依赖倒置原则,老司机还介绍了好几种其他系统构架的小技巧,推荐阅读。)
最后:在container,虚拟化,微服务,甚至serverless,FaaS盛行的今天,几百行code就可以是一个service,遵循依赖倒置原则来设计稳定的易拓展的系统,跟代码行数超过多少行有什么必然联系么???
====== 小尾巴 =====
我是阿莱克西斯,10+年技术经验关于高并发,大数据,与内部科学家合作搭建预测优化后段黑魔法系统的Amazon Sr. SDE,每天看技术书和paper到夜里1点的读书狂魔兼猫奴。
关注我,关注我的读书专栏,和解释分布式系统paper的专栏,带你成为程序艺术家,玩转分布式系统~
克劳备忘录也好,凯南电报也好,有两大共同点。首先,都是以现实主义的眼光去分析双方的关系。然后,给出的建议都是阳谋,并不是什么不可告人的阴谋,执行起来需要的不是鸡鸣狗盗的小聪明,而是惊人的意志力。
而美国现在战略界现实主义被边缘化,我推测,布热津斯基,基辛格那帮人应该写过不少。不过没所谓,美国能执行大战略的时代过去了。现在这一代精英上半年能管下半年就已经很了不起了。一个需要两代人以上持之以恒去完成的大战略,搞出来他们也执行不了。
冷战时期,从杜鲁门艾森豪威尔到肯尼迪尼克松,最后到李根老布什,个人性格和政治偏好差距不要太大,但是都忠实地完成了他们历史任务,沿着围堵政策做下去。这种战略定力和延续性,世间少见。在中国领导集团上能看见一些相似的东西,但是我们离得距离太近,反而看不清。但在美国精英层身上完全看不到这一点。
个人愚见。
这个4年前的问题选择在这个时间点突然出现在我今天的时间线上显得非常 亦可赛艇!
Android是2008年初才发布,而Oracle在2009年就以7.4B$收购了Sun,是Google不够睿智吗?
非也!
1)如果Android没有如此成功,Java对于Google而言就是一坨shit,Google从来没有想到自己会站在一坨翔上面取得空前的成功,如果有算命的告诉Google的命中贵人是阿翔,它就是穿越回去吃也要把它吃下去,可惜历史不能假设!
2)Google一直有python基因,很多系统都是基于python的,你知道工程师主导文化的可怕性吗?这帮pythonic的nerd出于情怀或者节操或者叫清高或者叫偏执或者叫真爱,它说什么都不会去买Java的,“老子看不上”!谁知造化弄人,09年你对我爱答不理,18年老子叫你高攀不起88亿!(注:今天的Google在各种收购之后,Java服务的比重占的也非常大了,变成了一个杂合的技术栈,而官司也很可能打到高院,尚未定论)
3)Google一直有跟开源保持共存共荣共襄盛举的传统,它跟Mozilla做生意,赞助开源项目,捐赠Wiki,主张“不作恶”,简直就是一副乌托邦理想主义者的化身,圈粉无数(包含答主),像Java这种项目,它更可能的方式是烧一笔钱给它花,然后来几句“希望Java明天会更好”之类的废话,它根本就不曾想过有一个家伙抄底了,因为那时候Android根本就没有火,Google从来就没有想过Java也T-M-D算哪门子“底”?
4)Sun的主手人也是个技术型的,就是技术牛掰业务做的稀烂,当时怎么看Sun都处在夕阳,SPARC也是逼格满满业务下滑被Intel捣的稀烂,那个价格没有几家觉得划算的,幸好是Oracle这种剑走偏锋的收购了它,要是换一家公司收购多半就把Sun雪藏甚至捣腾碎了,Java也就没有今日风光了,而Google在坊间也有创业公司杀手的美称,也许这就已经是历史发展的最好结果了。
什么,你问我对于Oracle收购Sun和MySQL怎么看?
还能怎么看?好白菜都让猪给拱呢呗!
但是作为吃瓜群众,我最喜欢看大佬们掐架,Google与Oracle的这场官司绝对酸爽,大家保持关注,各家都有千百号律师,吵起架来想想都 亦可赛艇!学知识产权法/专利法/法理学的同学们千万不要错过,说不定两年后就能进教材作案例呢!
什么,你又问我Google应该怎么做?
靠,我有不是劈柴!按我的观点,Google这次是违反了Java的使用协议的(无意引战,定论的事情留给专业法官),不能因为体量大就以为能压死人,那可是在美帝,万事全靠律师一张嘴,怎么讲都有理!
大家还记得微软以前有个skydrive吗?在英国被判败诉了,最后也得改名叫OneDrive呢!Google有钱了不起啊,过来领罚单!
而Java的坑早早就埋在那里了,所以苹果直接一刀切:老子不支持,免得搞一嘴毛!Flash一身毛病,一刀切,老子不支持!
所以,我对Google的建议是:
这TM不是关乎技术,不是关乎信仰,不是关乎生态,不是关乎用户体验!
这TM关系到命!
什么?要我预测结果?
法官中间调停,你们俩和解,google把赚的钱按每部手机给Oracle付钱?什么你说太扯了?你每买一部Android,都要给微软钱,你造吗?Oracle就想躺着就把钱收了!