大数据Scala编程.问题集(03)

大数据Scala编程.问题集(03)


by 高焕堂


高老师陪您成长...

  wKioL1XP7GXQd_ZbAAEShxXc7cA678.jpg 


Q-03: 如何从TemplateMethod设计模式理解trait的两项接口?

Answer:

    Scala的trait就是接口类(Interface class),其意味着,它的结构是抽象类,但其扮演接口的角色。通常,可从有两个不同视角去看待trait机制:

第一个视角:将trait对应到Java的"Interface"机制。此时,将trait看成一个接口(主要是抽象函数)加上一些具象函数;或者说接口里的某些函数具有实现代码。这是一般具有Java经验的开发者所孰悉的视角。 

第二个视角:将tarit对应到 TemplateMethod设计模式。此时,将trait看成一个接口类,其定义了两项接口。具象函数对应到 TemplateMethod模式的TemplateMethod函数,成为提供给Client的服务接口(Client interface,简称CI)。并且将抽象函数对应到PrimitiveOperation()函数,成为控制子类(Subclass)的扩充接口(Extends interface,简称<I>)。如下图所示:


    wKiom1SQZzrCfV0MAACkbcb3Ef8330.jpg

    (GoF的Design Patterns一书里的TemplateMethod模式)


为什么上述第二个视角会受到(架构)设计师的重视呢? 兹先从简单的Client/Server架构说起吧,如下图:


wKioL1SQZ9vBnTrvAABcsJ9bBkw802.jpg

由于这个Client调用了多个不同的接口(如图里的IS-1、IS-2 ... IS-n等),不仅导致Client模块会变得很复杂,而且这些接口的变动都会导致Client模块的变动,会大幅提升Server模块(如Server-1、Server-2 ... Server-n等)的变动(更新或改版)的成本。所以这种架构是不好的。


wKiom1SQZzzw9AElAABrs0Zq8f8061.jpg

Gamma等专家在Design Patterns一书里提出了Facade模式来化解上述的问题。如下图:


    wKioL1SQZ9yRBSZJAABqBdS98Mg569.jpg

Facade模式将Server端的众多接口(如图里的IS-1、IS-2 ... IS-n等)封装起来,转而提供单一接口(如IF接口)给Client来使用。这个Facade模快就能落实为Scala的trait机制。如下图:


    wKiom1SQZz7wUbRcAAB-aLmgg4k719.jpg

如果将这个trait模块取名为TFacade,可表示如下图:


 wKioL1SQaMKRnaueAABsyI5CBGw270.jpg

   看起来,这个TFacade模块已经化解了上述的问题了。但是,从(架构)设计师的角度来看,经常会问一个议题,就是:谁来设计或开发这个trait模块呢?如下图:


    wKiom1SQaCOz-ysiAACoB4d4kco239.jpg

因为TFacade掌控了从Client到Server之间的数据流动接口(通道)。从架构设计和分工的的角度来看,谁负责设计(或开发)这个trait模块,谁就掌握了这个信息流动接口(通道)。


wKiom1SQaCWAOz7fAACwDvbeynA118.jpg

为了让你看出这tarit模块所掌握的接口,我们将它配成TemplateMethod模式,如下图:


wKioL1SQaMbQ7QpMAACcDLT2ToA821.jpg

     TFacade里的具象函数,成为对Cleint的服务接口,如下图:


    wKiom1SQaRfTdL4-AACbkEB-AnU704.jpg

     TFacade里的抽象函数,成为与子类对接的接口,如下图:

    

    wKioL1SQafrCfhnwAACTl06Z41U287.jpg

仔细看看这TFacade模块内部的具象和抽象函数。可以看出来这TFacade模块的设计者(或开发者),他设计(或定义)了两项接口:

CI接口:由TFacade的具象函数所构成的接口。它由TFacade来实现,让Client来享用。

<I>接口:由TFacade的抽象函数所构成的接口。它是由TFacade(的开发者)所定义,让它的子类来实现服务于Client;而抽象函数(以<I>表示)则用来控制(然后服务)子类。而子类则实践了这个<I>接口(里的抽象函数)。如下图:


    wKioL1SQafyiikKDAAChSi0OLO8152.jpg

这两个接口有其微妙而重要的差别:对TFacade模块而言,CI是被动的接口;而<I>则是主动的接口。CI是由TFacade来实现,让Client来主动调用,TFacade是被动的提供服务,所以称为TFacade的被动型接口。反之,对TFacade模块而言,<I>则是主动的接口;<I>是TFacade(的开发者)定义的,由其子类所实现,让TFacade来主动调用子类提供的服务,所以称为TFacade的主动型接口。

我(高焕堂)从上述的TempalteMethod和Facade等设计模式,更进一步的进行抽象化,而得出更抽象、更通用的的设计模式,特别称为"EIT造形",如下图:


wKiom1SQaVyTj6PsAAC7tg4fUWg474.jpg 14

这TFacade(即trait)模块里的具象函数里的代码,担任信息流动的逻辑控制部分,相当于汽车的引擎功能,所以称为<E>部分。这TFacade模块里的抽象函数,担任这<E>与子类的接口部分,子类相当于汽车的轮胎部分,所以称为<T>部分。<I>则是引擎与轮胎的接口部分。


wKioL1SQaf3jah4PAACfOOMW5No270.jpg

        这就是一个EIT造形的架构了。


    wKiom1SQaV6DC-L3AACXgQ9To1A522.jpg

     关于这EIT造形的<E&I>(可落实为trait机制)的知识及其妙用技术,请您观看

我的:<<EIT架构设计系列视频课程>>


请继续学习高老��的相�P��l

相关文章:大数据Scala编程问题集(01)

相关文章:大数据Scala编程问题集(02)


~ End ~


你可能感兴趣的:(scala,EIT,高�ㄌ�)