22 | 理论八:如何用迪米特法则(LOD)实现“高内聚、松耦合”?
下载APP
关闭
渠道合作
推荐作者
22 | 理论八:如何用迪米特法则(LOD)实现“高内聚、松耦合”?
2019-12-23 王争 来自北京
《设计模式之美》
课程介绍
讲述:冯永吉
时长15:33大小14.24M
今天,我们讲最后一个设计原则:迪米特法则。尽管它不像 SOLID、KISS、DRY 原则那样,人尽皆知,但它却非常实用。利用这个原则,能够帮我们实现代码的“高内聚、松耦合”。今天,我们就围绕下面几个问题,并结合两个代码实战案例,来深入地学习这个法则。
什么是“高内聚、松耦合”?
如何利用迪米特法则来实现“高内聚、松耦合”?
有哪些代码设计是明显违背迪米特法则的?对此又该如何重构?
话不多说,让我们开始今天的学习吧!
何为“高内聚、松耦合”?
“高内聚、松耦合”是一个非常重要的设计思想,能够有效地提高代码的可读性和可维护性,缩小功能改动导致的代码改动范围。实际上,在前面的章节中,我们已经多次提到过这个设计思想。很多设计原则都以实现代码的“高内聚、松耦合”为目的,比如单一职责原则、基于接口而非实现编程等。
实际上,“高内聚、松耦合”是一个比较通用的设计思想,可以用来指导不同粒度代码的设计与开发,比如系统、模块、类,甚至是函数,也可以应用到不同的开发场景中,比如微服务、框架、组件、类库等。为了方便我讲解,接下来我以“类”作为这个设计思想的应用对象来展开讲解,其他应用场景你可以自行类比。
在这个设计思想中,“高内聚”用来指导类本身的设计,“松耦合”用来指导类与类之间依赖关系的设计。不过,这两者并非完全独立不相干。高内聚有助于松耦合,松耦合又需要高内聚的支持。
那到底什么是“高内聚”呢?
所谓高内聚,就是指相近的功能应该放到同一个类中,不相近的功能不要放到同一个类中。相近的功能往往会被同时修改,放到同一个类中,修改会比较集中,代码容易维护。实际上,我们前面讲过的单一职责原则是实现代码高内聚非常有效的设计原则。对于这一点,你可以回过头再去看下专栏的第 15 讲。
我们再来看一下,什么是“松耦合”?
所谓松耦合是说,在代码中,类与类之间的依赖关系简单清晰。即使两个类有依赖关系,一个类的代码改动不会或者很少导致依赖类的代码改动。实际上,我们前面讲的依赖注入、接口隔离、基于接口而非实现编程,以及今天讲的迪米特法则,都是为了实现代码的松耦合。
最后,我们来看一下,“内聚”和“耦合”之间的关系。
前面也提到,“高内聚”有助于“松耦合”,同理,“低内聚”也会导致“紧耦合”。关于这一点,我画了一张对比图来解释。图中左边部分的代码结构是“高内聚、松耦合”;右边部分正好相反,是“低内聚、紧耦合”。
图中左边部分的代码设计中,类的粒度比较小,每个类的职责都比较单一。相近的功能都放到了一个类中,不相近的功能被分割到了多个类中。这样类更加独立,代码的内聚性更好。因为职责单一,所以每个类被依赖的类就会比较少,代码低耦合。一个类的修改,只会影响到一个依赖类的代码改动。我们只需要测试这一个依赖类是否还能正常工作就行了。
图中右边部分的代码设计中,类粒度比较大,低内聚,功能大而全,不相近的功能放到了一个类中。这就导致很多其他类都依赖这个类。当我们修改这个类的某一个功能代码的时候,会影响依赖它的多个类。我们需要测试这三个依赖类,是否还能正常工作。这也就是所谓的“牵一发而动全身”。
除此之外,从图中我们也可以看出,高内聚、低耦合的代码结构更加简单、清晰,相应地,在可维护性和可读性上确实要好很多。
“迪米特法则”理论描述
迪米特法则的英文翻译是:Law of Demeter,缩写是 LOD。单从这个名字上来看,我们完全猜不出这个原则讲的是什么。不过,它还有另外一个更加达意的名字,叫作最小知识原则,英文翻译为:The Least Knowledge Principle。
关于这个设计原则,我们先来看一下它最原汁原味的英文定义:
Each unit should have only limited knowledge about other units: only units “closely” related to the current unit. Or: Each unit should only talk to its friends; Don’t talk to strangers.
我们把它直译成中文,就是下面这个样子:
每个模块(unit)只应该了解那些与它关系密切的模块(units: only units “closely” related to the current unit)的有限知识(knowledge)。或者说,每个模块只和自己的朋友“说话”(talk),不和陌生人“说话”(talk)。
我们之前讲过,大部分设计原则和思想都非常抽象,有各种各样的解读,要想灵活地应用到实际的开发中,需要有实战经验的积累。迪米特法则也不例外。所以,我结合我自己的理解和经验,对刚刚的定义重新描述一下。注意,为了统一讲解,我把定义描述中的“模块”替换成了“类”。
不该有直接依赖关系的类之间,不要有依赖;有依赖关系的类之间,尽量只依赖必要的接口(也就是定义中的“有限知识”)。
从上面的描述中,我们可以看出,迪米特法则包含前后两部分,这两部分讲的是两件事情,我用两个实战案例分别来解读一下。
理论解读与代码实战一
我们先来看这条原则中的前半部分,“不该有直接依赖关系的类之间,不要有依赖”。我举个例子解释一下。
这个例子实现了简化版的搜索引擎爬取网页的功能。代码中包含三个主要的类。其中,NetworkTransporter 类负责底层网络通信,根据请求获取数据;HtmlDownloader 类用来通过 URL 获取网页;Document 表示网页文档,后续的网页内容抽取、分词、索引都是以此为处理对象。具体的代码实现如下所示:
这段代码虽然“能用”,能实现我们想要的功能,但是它不够“好用”,有比较多的设计缺陷。你可以先试着思考一下,看看都有哪些缺陷,然后再来看我下面的讲解。
首先,我们来看 NetworkTransporter 类。作为一个底层网络通信类,我们希望它的功能尽可能通用,而不只是服务于下载 HTML,所以,我们不应该直接依赖太具体的发送对象 HtmlRequest。从这一点上讲,NetworkTransporter 类的设计违背迪米特法则,依赖了不该有直接依赖关系的 HtmlRequest 类。
我们应该如何进行重构,让 NetworkTransporter 类满足迪米特法则呢?我这里有个形象的比喻。假如你现在要去商店买东西,你肯定不会直接把钱包给收银员,让收银员自己从里面拿钱,而是你从钱包里把钱拿出来交给收银员。这里的 HtmlRequest 对象就相当于钱包,HtmlRequest 里的 address 和 content 对象就相当于钱。我们应该把 address 和 content 交给 NetworkTransporter,而非是直接把 HtmlRequest 交给 NetworkTransporter。根据这个思路,NetworkTransporter 重构之后的代码如下所示:
我们再来看 HtmlDownloader 类。这个类的设计没有问题。不过,我们修改了 NetworkTransporter 的 send() 函数的定义,而这个类用到了 send() 函数,所以我们需要对它做相应的修改,修改后的代码如下所示:
最后,我们来看下 Document 类。这个类的问题比较多,主要有三点。第一,构造函数中的 downloader.downloadHtml() 逻辑复杂,耗时长,不应该放到构造函数中,会影响代码的可测试性。代码的可测试性我们后面会讲到,这里你先知道有这回事就可以了。第二,HtmlDownloader 对象在构造函数中通过 new 来创建,违反了基于接口而非实现编程的设计思想,也会影响到代码的可测试性。第三,从业务含义上来讲,Document 网页文档没必要依赖 HtmlDownloader 类,违背了迪米特法则。
虽然 Document 类的问题很多,但修改起来比较简单,只要一处改动就可以解决所有问题。修改之后的代码如下所示:
理论解读与代码实战二
现在,我们再来看一下这条原则中的后半部分:“有依赖关系的类之间,尽量只依赖必要的接口”。我们还是结合一个例子来讲解。下面这段代码非常简单,Serialization 类负责对象的序列化和反序列化。提醒你一下,有个类似的例子在之前的第 15 节课中讲过,你可以结合着一块儿看一下。
单看这个类的设计,没有一点问题。不过,如果我们把它放到一定的应用场景里,那就还有继续优化的空间。假设在我们的项目中,有些类只用到了序列化操作,而另一些类只用到反序列化操作。那基于迪米特法则后半部分“有依赖关系的类之间,尽量只依赖必要的接口”,只用到序列化操作的那部分类不应该依赖反序列化接口。同理,只用到反序列化操作的那部分类不应该依赖序列化接口。
根据这个思路,我们应该将 Serialization 类拆分为两个更小粒度的类,一个只负责序列化(Serializer 类),一个只负责反序列化(Deserializer 类)。拆分之后,使用序列化操作的类只需要依赖 Serializer 类,使用反序列化操作的类只需要依赖 Deserializer 类。拆分之后的代码如下所示:
不知道你有没有看出来,尽管拆分之后的代码更能满足迪米特法则,但却违背了高内聚的设计思想。高内聚要求相近的功能要放到同一个类中,这样可以方便功能修改的时候,修改的地方不至于过于分散。对于刚刚这个例子来说,如果我们修改了序列化的实现方式,比如从 JSON 换成了 XML,那反序列化的实现逻辑也需要一并修改。在未拆分的情况下,我们只需要修改一个类即可。在拆分之后,我们需要修改两个类。显然,这种设计思路的代码改动范围变大了。
如果我们既不想违背高内聚的设计思想,也不想违背迪米特法则,那我们该如何解决这个问题呢?实际上,通过引入两个接口就能轻松解决这个问题,具体的代码如下所示。实际上,我们在第 18 节课中讲到“接口隔离原则”的时候,第三个例子就使用了类似的实现思路,你可以结合着一块儿来看。
尽管我们还是要往 DemoClass_1 的构造函数中,传入包含序列化和反序列化的 Serialization 实现类,但是,我们依赖的 Serializable 接口只包含序列化操作,DemoClass_1 无法使用 Serialization 类中的反序列化接口,对反序列化操作无感知,这也就符合了迪米特法则后半部分所说的“依赖有限接口”的要求。
实际上,上面的的代码实现思路,也体现了“基于接口而非实现编程”的设计原则,结合迪米特法则,我们可以总结出一条新的设计原则,那就是“基于最小接口而非最大实现编程”。有些同学之前问,新的设计模式和设计原则是怎么创造出来的,实际上,就是在大量的实践中,针对开发痛点总结归纳出来的套路。
辩证思考与灵活应用
对于实战二最终的设计思路,你有没有什么不同的观点呢?
整个类只包含序列化和反序列化两个操作,只用到序列化操作的使用者,即便能够感知到仅有的一个反序列化函数,问题也不大。那为了满足迪米特法则,我们将一个非常简单的类,拆分出两个接口,是否有点过度设计的意思呢?
设计原则本身没有对错,只有能否用对之说。不要为了应用设计原则而应用设计原则,我们在应用设计原则的时候,一定要具体问题具体分析。
对于刚刚这个 Serialization 类来说,只包含两个操作,确实没有太大必要拆分成两个接口。但是,如果我们对 Serialization 类添加更多的功能,实现更多更好用的序列化、反序列化函数,我们来重新考虑一下这个问题。修改之后的具体的代码如下:
在这种场景下,第二种设计思路要更好些。因为基于之前的应用场景来说,大部分代码只需要用到序列化的功能。对于这部分使用者,没必要了解反序列化的“知识”,而修改之后的 Serialization 类,反序列化的“知识”,从一个函数变成了三个。一旦任一反序列化操作有代码改动,我们都需要检查、测试所有依赖 Serialization 类的代码是否还能正常工作。为了减少耦合和测试工作量,我们应该按照迪米特法则,将反序列化和序列化的功能隔离开来。
重点回顾
好了,今天的内容到此就讲完了。我们一块来总结回顾一下,你需要掌握的重点内容。
1. 如何理解“高内聚、松耦合”?
“高内聚、松耦合”是一个非常重要的设计思想,能够有效提高代码的可读性和可维护性,缩小功能改动导致的代码改动范围。“高内聚”用来指导类本身的设计,“松耦合”用来指导类与类之间依赖关系的设计。
所谓高内聚,就是指相近的功能应该放到同一个类中,不相近的功能不要放到同一类中。相近的功能往往会被同时修改,放到同一个类中,修改会比较集中。所谓松耦合指的是,在代码中,类与类之间的依赖关系简单清晰。即使两个类有依赖关系,一个类的代码改动也不会或者很少导致依赖类的代码改动。
2. 如何理解“迪米特法则”?
不该有直接依赖关系的类之间,不要有依赖;有依赖关系的类之间,尽量只依赖必要的接口。迪米特法则是希望减少类之间的耦合,让类越独立越好。每个类都应该少了解系统的其他部分。一旦发生变化,需要了解这一变化的类就会比较少。
课堂讨论
在今天的讲解中,我们提到了“高内聚、松耦合”“单一职责原则”“接口隔离原则”“基于接口而非实现编程”“迪米特法则”,你能总结一下它们之间的区别和联系吗?
欢迎在留言区写下你的答案,和同学一起交流和分享。如果有收获,也欢迎你把这篇文章分享给你的朋友。
分享给需要的人,Ta购买本课程,你将得29元
生成海报并分享
赞 121
提建议
© 版权归极客邦科技所有,未经许可不得传播售卖。 页面已增加防盗追踪,如有侵权极客邦将依法追究其法律责任。
上一篇
21 | 理论七:重复的代码就一定违背DRY吗?如何提高代码的复用性?
下一篇
23 | 实战一(上):针对业务系统的开发,如何做需求分析和设计?
精选留言(123)
- 知行合一2019-12-23目的都是实现高内聚低耦合,但是出发的角度不一样,单一职责是从自身提供的功能出发,迪米特法则是从关系出发,针对接口而非实现编程是使用者的角度,殊途同归。共 7 条评论368
- 下雨天2019-12-251.单一职责原则 适用对象:模块,类,接口 侧重点:高内聚,低耦合 思考角度:自身 2.接口隔离原则 适用对象:接口,函数 侧重点:低耦合 思考角度:调用者 3.基于接口而非实现编程 适用对象:接口,抽象类 侧重点:低耦合 思考角度:调用者 4.迪米特法则 适用对象:模块,类 侧重点:低耦合 思考角度:类关系展开共 12 条评论225
- Ken张云忠2019-12-23“高内聚、松耦合”“单一职责原则”“接口隔离原则”“基于接口而非实现编程”“迪米特法则”,它们之间的区别和联系吗? 区别: 高内聚、松耦合:是一个重要的设计思想,能够有效地提高代码的可读性和可维护性,缩小功能改动导致的代码改动范围. 单一职责原则:A class or module should have a single reponsibility.提供的功能上要单一. 接口隔离原则:Clients should not be forced to depend upon interfaces that they do not use.与外部关系上只依赖需要的抽象. 基于接口而非实现编程:Program to an interface, not an implementation.是一条比较抽象、泛化的设计思想,为了提高代码的灵活性/扩展性/可维护性. 迪米特法则:Each unit should have only limited knowledge about other units: only units “closely” related to the current unit. Or: Each unit should only talk to its friends; Don’t talk to strangers.每个单元只该依赖与它关系密切的单元,最少知道,只与关系密切的单一交互. 联系: 职责越单一越容易做到接口隔离,也越容易做到最少知道的迪米特法则. 基于抽象编程抽象的知识越顶层越脱离具体实现,相对知道的内容就越少,也容易实现迪米特法则. 接口隔离原则与迪米特法则都强调只依赖需要的部分,接口隔离原则是相对偏上层来说的,迪米特法则是相对偏具体实现来说的. 单一职责原则/接口隔离原则/基于接口而非实现编程/迪米特法则都以实现代码的"高内聚、松耦合"为目的,提高代码的可读性和可维护性,缩小功能改动导致的代码改动范围,降低风险.展开58
- 辣么大2019-12-23关于LoD,请记住一条:方法中不要使用ChainMethods。 坏的实践: Amount = customer.orders().last().totals().amount() 和 orders = customer.orders() lastOders = orders.last() totals = lastOders.totals() amount = totals.amount() 上面的例子中,chain中的方法改变会影响很多地方。这里注意区别建造者模式和pipeline管道,这两种的chain中的方法不易改变。 出现这样的代码,需要考虑可能是设计或实现出了问题。 LoD如何使用: 一个类C中的方法只能调用: 1、C中其他实例方法 2、它自己的参数方法 3、它创建对象的方法 4、不要调用全局变量(包括可变对象、可变单例) 例如: class HtmlDownloader{ Html html; public void downloadHtml(Transporter trans, String url){ if(checkUrl(url)){// ok 自己的实例方法 // return } rawData = trans.send(uri);// ok 参数对象的方法 Html html = createHtml(rawData); // ok 它创建的对象 html.save();// ok 它创建对象的方法 ) private boolean checkUrl(String url){ // check } } 参考: The Pragmatic Programmer 1st edition and 2nd edition展开共 24 条评论40
- xzy2020-02-20看一遍,真的不行,每次看都有新收获!17
- LiuHu2019-12-23“高内聚、松耦合” 是衡量好代码的标准之一,为了实现这样的目标,我们需要遵循如下原则: “基于接口而非实现编程”,接口本身就是一层抽象,接口是稳定的,实现是易变的,强调的是基于契约编程,这样能够隔离变化。实现细节代码的变化,不影依赖该接口的对象,到从而达到松耦合的目的。 “迪米特法则”,定义的是发布的接口(类、模块等)能不能依赖,如何依赖的问题。使用者去除不必要的依赖,只依赖必要的接口。这样即使接口一旦发生变化,需要了解这一变化的类就会比较少,达到松耦合的目的。 “接口隔离原则”,从使用者的角度考虑如何设计接口,让使用者只依赖必要的接口,不会被迫依赖不用的接口。这样即使接口一旦发生变化,需要了解这一变化的类就会比较少,这样就能符合 “迪米特法则” 。 “单一职责原则”,针对模块、类、接口的设计,将功能相关性很强的代码抽取到一起,达到高内聚的目标。展开15
- 王大喵2020-11-16联系: “接口隔离原则”是客户端不应该被强迫依赖不需要的接口,和“迪米特法则”中的有限知识异曲同工,接口簇会更加“单一职责”实现方式“基于接口而非实现编程”,达到的目的是高内聚,松耦合。 区别: 1. 各种原则最终的目的是为了实现“高内聚、松耦合”。 2. 单一职责原则 主要是指导类和模块,避免大而全,提高内聚性。 3. 接口隔离和迪米特(最小知识)主要指导“松耦合”,解耦使用方的依赖。 4. 基于接口而非实现编程:主要是解耦接口和实现,是指导思想,提高扩展性。展开
作者回复: ������
共 2 条评论8 - 落叶飞逝的恋2019-12-23课后讨论:代码的最终目的是高内聚、松耦合的。而为了达到这个目的,就需要利用到迪米特法则。而迪米特法则的实现,又需要利用单一职责将单个类定义职责单一化,并且为了解决多个类之间的关系,又需要用到基于接口编程而非实现编程。这样类与类之间就相当于契约化,也就是不关心类的具体实现。7
- 郑大钱2020-09-07一不小心就能用到一个牛逼的原则,如果没有这些归纳总结的话,永远都不知道自己究竟有多牛逼。 是不是觉得迪米特法则和之前的原则很像,当然很像了,毕竟他们都是为了实现“高内聚,低耦合”。 【单一职责原则】是从自身功能出发,实现高内聚,低耦合 【接口隔离原则】和【基于接口而非实现编程】是从调用者出发,实现低耦合 【迪米特法则 】是从关系出发,实现低耦合展开4
- 提姆2020-07-07老师您好,请问一下Document的修改为什么会使用到Factory的方式去产生?我对这一步的修改没有很大的感受,我这里的认知是Document只是单纯对文件的操作,那是不是可以透过HtmlDownloader实现相关取得文件的接口,像是IDownloader之类的名称,并直接回传Document这个类(亦或者对此类做其它的延伸),未来也可以实现其它像是json等其他格式的downloader,不知道我这个想法是不是可行?
作者回复: 你的设计也可以的。设计本身没有最优解,合理、能自圆其说就可以了。总体上来讲,我这个例子是为了展示尽量减少类之间的耦合。
共 2 条评论3 - 小刀2019-12-28高内聚-低耦合 单一职责-自身功能出发 迪米特法则 类与类之间关系 接口而非实现编程-依赖必要的抽象3
- 戒惜舍得2019-12-27相近的功能。 怎么算相近啊。学晕了。
作者回复: 咋这么能抬杠呢 你说我问你怎么相近 你能回答上来吗 这个也没法量化 设计模式不是算法 什么都能定量 更多的是教你思想 你自己去感受
共 2 条评论3 - QQ怪2019-12-24老师讲的真棒👍3
- 岁月2019-12-23如果说高内聚、松耦合等价于"中国特色社会主义", 那么“单一职责原则”“接口隔离原则”“基于接口而非实现编程”“迪米特法则”这几个原则就像是在说如何才能做到做到这样的社会? 答案就是我们要"倡导富强、民主、文明、和谐, 自由、平等、公正、法治,爱国、敬业、诚信、友善"共 1 条评论3
- Geek_8095612020-11-08单一职责原则:模块,类,方法等功能要单一。功能越单一的类,对外部依赖越少。高内聚。 接口隔离原则: 调用者调用被调用者的方法或功能的时候,这个方法或者功能的粒度要小,职责要单一。高内聚 单一职责和接口隔离的区别是:一个是偏向于整体的(类,模块,接口的)设计,而接口隔离知识对接口的设计。相同点都是要求单一职责,高内聚。 基于接口而非实现编程:将接口和实现分离。调用者只需要知道被调用者抽象出来的方法或功能,并不需要知道具体的细节是怎么执行的。降低了调用者和被调用者之间的耦合度。更偏向于整体的抽象思想。 接口隔离与基于接口而非实现编程:一个是对具体接口的要求,(单一),一个是要求将具体的行为和功能抽象化,为后续被调用的整体替换做准备。 迪米特法则:又叫做最少知识法则,降低调用者与被调用者之间的耦合度。调用者只关于自己必须知道的知识(没有就实现不了的)。展开2
- 岁月2020-05-13感觉NetworkTransporter这个例子举的不好... 本身获取HTML确实可以通过通过HtmlRequest, 现在非得把他拆成一个地址和content, 那么地址和content分别传入什么呢? 这样做HtmlRequest这个类就没有存在的意义了. 下面document例子我比较认可. 确实document类只要把html字符串传进来即可, 至于谁去加载字符串这个不归他管共 2 条评论2
- KK2020-04-05对LOD迪米特法则的形象理解:不要探究朋友的隐私,要与陌生人保持距离。😂2
- ismind2019-12-27前几天没抽出时间,今天补上。 1,感觉还是要做笔记,现在才知道要做一下笔记,不然前面学的基本上只是留个大概的印象,效率低 2,感觉开始有难度了,综合性更高了,开始加速了么?哈哈 课后思考: 1,这五个原则与思想,联系就是:都是为了提高代码的可读性、扩展性、灵活性等,这也是这些原则产生的原因之一,我觉得应该把握这个主要的目的,再去应用这些原则或者设计模式,否则单纯为了使用这些原则,那很可能适得其反,我又想起那句话:“不忘初心方得始终”,学习设计模式也要这样吧。 老师的代码实战二,简单的序列化使用LOD原则,感觉过度设计了,不过一看后面功能变复杂了,觉得设计才合理。 2,区别:应该是关注点不同吧,“高内聚、低耦合”关注的是类,也就是考虑类与类之间的关系;单一职责则是关注前几天没抽出时间,今天补上。 1,感觉还是要做笔记,现在才知道要做一下笔记,不然前面学的基本上只是留个大概的印象,效率低 2,感觉开始有难度了,综合性更高了,开始加速了么?哈哈 课后思考: 1,这五个原则与思想,联系就是:都是为了提高代码的可读性、扩展性、灵活性等,这也是这些原则产生的原因之一,我觉得应该把握这个主要的目的,再去应用这些原则或者设计模式,否则单纯为了使用这些原则,那很可能适得其反,我又想起那句话:“不忘初心方得始终”,学习设计模式也要这样吧。 老师的代码实战二,简单的序列化使用LOD原则,感觉过度设计了,不过一看后面功能变复杂了,觉得设计才合理。 2,区别:应该是关注点不同吧,“高内聚、低耦合”关注的是类,也就是考虑类与类之间的关系;单一职责则是关注某一个类或者函数的功能是否单一;接口隔离则关注不要依赖不需要的接口;基于接口而非实现编程就考虑将实现替换为接口。 自己感觉这些还没有彻底理解,只是大概知道,感觉离掌握与熟悉还有一段距离。展开2
- 黄林晴2019-12-23打卡 所有的设计原则都相辅相成2
- Z宇锤锤2021-11-23高内聚,低耦合就是分门别类别掺和。 迪米特法则就是别和陌生人说话,只和熟人说话。调用过程中应该尽量解耦,尽量考虑使用。参考rule of three1