APP下载

你写的程式码是别人的噩梦吗?看阿里架构师从领域建模的必要性谈起

消息来源:baojiabao.com 作者: 发布时间:2024-04-28

报价宝综合消息你写的程式码是别人的噩梦吗?看阿里架构师从领域建模的必要性谈起

目录

为什么要领域建模DDD革命DDD初体验银行转账领域模型实现领域事件(Domain Event)领域服务(Domain Service)业务视觉化和可配置化作者简介:张建飞,Frank,阿里高阶技术专家。

文章转载自阿里技术公众号

为什么要领域建模

维护过企业级业务系统的同学,基本上没有一个不抱怨业务程式码烂的,过程式的面条程式码充斥着屏幕,程序员的心力和体力都经受着极大的考验,怎么破?

欢迎关注笔者,优质文章都在这里等你。

DDD革命

DDD革命性在于,领域模型准确反映了业务语言,而传统J2EE或Spring+Hibernate等事务性程式设计模型只关心资料,这些资料物件除了简单setter/getter方法外,没有任何业务方法,被比喻成贫血模式。

以银行账号Account为案例,Account有“存款”,“计算利息”和“取款”等业务行为,但是传统经典的方式是将“存款”,“计算利息”和“取款”行为放在账号的服务AccountService中,而不是放在Account物件本身之中。

我们不能因为用了计算机,用了数据库,用了框架,业务模型反而被技术框架给绑架了,就像人虽然是由母亲生的,但是人的吃喝拉撒母亲不能替代,更不能以母爱名义剥夺人的正常职责行为,如果是这样,这个人就是被母爱绑架了。

DDD不是银弹

软件的世界里没有银弹,是用事务指令码还是领域模型没有对错之分,关键看是否合适。就像自营和平台哪个模式更好?答案是都很好,所以亚马逊可以有三方入住,阿里也可以有自建仓嘛。

实际上,CQRS就是对事务指令码和领域模型两种模式的综合,因为对于Query和报表的场景,使用领域模型往往会把简单的事情弄复杂,此时完全可以用奥卡姆剃刀把领域层剃掉,直接访问Infrastructure。

我个人也是坚决反对过度设计的,因此对于简单业务场景,我强力建议还是使用事务指令码,其优点是简单、直观、易上手。但对于复杂的业务场景,你再这么玩就不行了,因为一旦业务变得复杂,事务指令码就很难应对,容易造成程式码的“一锅粥”,系统的腐化速度和复杂性呈指数级上升。

目前比较有效的治理办法就是领域建模,因为领域模型是面向物件的,在封装业务逻辑的同时,提升了物件的内聚性和重用性,因为使用了通用语言(Ubiquitous Language),使得隐藏的业务逻辑得到显性化表达,使得复杂性治理成为可能。

接下来,让我们看一个银行转账的例项,对比下事务指令码和领域模型两者程式设计模型的不同。

DDD初体验

银行转账事务指令码实现

在事务指令码的实现中,关于在两个账号之间转账的领域业务逻辑都被写在了MoneyTransferService的实现里面了,而Account仅仅是getters和setters的资料结构,也就是我们说的贫血模型:

public class MoneyTransferServiceTransactionScriptImpl

implements MoneyTransferService {

private AccountDao accountDao;

private BankingTransactionRepository bankingTransactionRepository;

. . .

@Override

public BankingTransaction transfer(

String fromAccountId, String toAccountId, double amount) {

Account fromAccount = accountDao.findById(fromAccountId);

Account toAccount = accountDao.findById(toAccountId);

. . .

double newBalance = fromAccount.getBalance() - amount;

switch (fromAccount.getOverdraftPolicy()) {

case NEVER:

if (newBalance throw new DebitException("Insufficient funds");

}

break;

case ALLOWED:

if (newBalance throw new DebitException(

"Overdraft limit (of " + limit + ") exceeded: " + newBalance);

}

break;

}

fromAccount.setBalance(newBalance);

toAccount.setBalance(toAccount.getBalance() + amount);

BankingTransaction moneyTransferTransaction =

new MoneyTranferTransaction(fromAccountId, toAccountId, amount);

bankingTransactionRepository.addTransaction(moneyTransferTransaction);

return moneyTransferTransaction;

}

}

上面的程式码大家看起来应该比较眼熟,因为目前大部分系统都是这么写的。需求评审完,工程师画几张UML图完成设计,就开始向上面这样怼业务程式码了,这样写基本不用太费脑,完全是面向过程的程式码风格。

有些同学可能会说,我这样写也可以实现系统功能啊,还是那句话“just because you can, doesn’t mean you should”。说句不好听的,正是有这么多“没有追求”、“不求上进”的码农才造成了应用系统的混乱、败坏了应用开发的名声。

这也是为什么很多应用开发工程师觉得工作没意思,技术含量低,觉得整天就是写if-else的业务逻辑程式码,系统又烂,工作繁琐、无聊、没有成长、没有成就感,所以转向去做中介软件啊,去写JDK啊,觉得那个NB。

实际上,应用开发一点都不简单也不无聊,业务的变化比底层Infrastructure的变化要多得多,解决的难度也丝毫不比写底层程式码容易,只是很多人选择了用无聊的方式去做。

其实我们是有办法做的更优雅的,这种优雅的方式就是领域建模,唯有掌握了这种优雅你才能实现从工程师向应用架构的转型。同样的业务逻辑,接下来就让我们看一下用DDD是怎么做的。

银行转账领域模型实现

如果用DDD的方式实现,Account实体除了账号属性之外,还包含了行为和业务逻辑,比如debit( )和credit( )方法。

// @Entity

public class Account {

// @Id

private String id;

private double balance;

private OverdraftPolicy overdraftPolicy;

. . .

public double balance() { return balance; }

public void debit(double amount) {

this.overdraftPolicy.preDebit(this, amount);

this.balance = this.balance - amount;

this.overdraftPolicy.postDebit(this, amount);

}

public void credit(double amount) {

this.balance = this.balance + amount;

}

}

而且透支策略OverdraftPolicy也不仅仅是一个Enum了,而是被抽象成包含了业务规则并采用了策略模式的物件。

public interface OverdraftPolicy {

void preDebit(Account account, double amount);

void postDebit(Account account, double amount);

}

public class NoOverdraftAllowed implements OverdraftPolicy {

public void preDebit(Account account, double amount) {

double newBalance = account.balance() - amount;

if (newBalance throw new DebitException("Insufficient funds");

}

}

public void postDebit(Account account, double amount) {

}

}

public class LimitedOverdraft implements OverdraftPolicy {

private double limit;

. . .

public void preDebit(Account account, double amount) {

double newBalance = account.balance() - amount;

if (newBalance throw new DebitException(

"Overdraft limit (of " + limit + ") exceeded: " + newBalance);

}

}

public void postDebit(Account account, double amount) {

}

}

而Domain Service只需要呼叫Domain Entity物件完成业务逻辑即可。

public class MoneyTransferServiceDomainModelImpl

implements MoneyTransferService {

private AccountRepository accountRepository;

private BankingTransactionRepository bankingTransactionRepository;

. . .

@Override

public BankingTransaction transfer(

String fromAccountId, String toAccountId, double amount) {

Account fromAccount = accountRepository.findById(fromAccountId);

Account toAccount = accountRepository.findById(toAccountId);

. . .

fromAccount.debit(amount);

toAccount.credit(amount);

BankingTransaction moneyTransferTransaction =

new MoneyTranferTransaction(fromAccountId, toAccountId, amount);

bankingTransactionRepository.addTransaction(moneyTransferTransaction);

return moneyTransferTransaction;

}

}

通过上面的DDD重构后,原来在事务指令码中的逻辑,被分散到Domain Service,Domain Entity和OverdraftPolicy三个满足SOLID的物件中,在继续阅读之前,我建议可以自己先体会一下DDD的好处。

领域建模的好处

DDD最大的好处是:接触到需求第一步就是考虑领域模型,而不是将其切割成资料和行为,然后资料用数据库实现,行为使用服务实现,最后造成需求的首肢分离。DDD让你首先考虑的是业务语言,而不是资料。DDD强调业务抽象和面向物件程式设计,而不是过程式业务逻辑实现。重点不同导致程式设计世界观不同。

面向物件

封装:Account的相关操作都封装在Account Entity上,提高了内聚性和可重用性。多型:采用策略模式的OverdraftPolicy(多型的典型应用)提高了程式码的可扩充套件性。业务语义显性化

通用语言:“一个团队,一种语言”,将模型作为语言的支柱。确保团队在内部的所有交流中,程式码中,画图,写东西,特别是讲话的时候都要使用这种语言。例如账号,转账,透支策略,这些都是非常重要的领域概念,如果这些命名都和我们日常讨论以及PRD中的描述保持一致,将会极大提升程式码的可读性,减少认知成本。说到这,稍微吐槽一下我们有些工程师的英语水平,有些神翻译让一些核心领域概念变得面目全非。显性化:就是将隐式的业务逻辑从一推if-else里面抽取出来,用通用语言去命名、去写程式码、去扩充套件,让其变成显示概念,比如“透支策略”这个重要的业务概念,按照事务指令码的写法,其含义完全淹没在程式码逻辑中没有突显出来,看程式码的人自然也是一脸懵逼,而领域模型里面将其用策略模式抽象出来,不仅提高了程式码的可读性,可扩充套件性也好了很多。如何进行领域建模

初步建模

好的模型应该是建立在对业务深入理解的基础上。就我自己的经验而言,建模是一个不断迭代的过程,一开始可以简单点来。

首先抓住一些核心概念,这些业务知识和核心概念可以通过和业务专家沟通,也可以通过头脑风暴的形式从User Story或者Event Storming去扣。

然后假设一些业务场景走查一下,再写一些虚拟码验证一下run一下,看看顺不顺,如果很顺滑,说明没毛病,否则就要看看是不是需要调整一下模型,随着专案的进行和对业务理解的不断深入,这种迭代将持续进行。

举个栗子,比如让你设计一个中介系统,一个典型的User Story可能是“小明去找工作,中介说你留个电话,有工作机会我会通知你”,这里面的关键名词很可能就是我们需要的领域物件:

- 小明是求职者。

- 电话是求职者的属性。

- 中介包含了中介公司,中介员工两个关键物件。

- 工作机会肯定也是关键领域物件;

- 通知这个动词暗示我们这里用观察者模式会比较合适。

然后再梳理一下领域物件之间的关系,一个求职者可以应聘多个工作机会,一个工作机会也可以被多个求职者应聘,M2M的关系,中介公司可以包含多个员工,O2M的关系。对于这样简单的场景,这个建模就差不多了。

当然我们的业务场景往往比这个要复杂,而且不是所有的名词都是领域物件也可能是属性,也不是所有的动词都是方法也可能是领域物件,再者,看的见实体好找,看不见的、隐藏的,需要深入理解业务,需要“无中生有”才能得到的抽象就没那么容易发现了。

所以要具体问题具体对待,这个进化的过程需要我们有很好的业务理解力,抽象能力以及建模的经验(知道为什么公司的job model里那么强调技术人员的业务理解力和抽象能力了吧。

比如通常情况下,价格和库存只是订单和商品的一个属性,但是在阿里系电商业务场景下,价格计算和库存扣减的复杂程度可以让你怀疑人生,因此作为电商中台,把价格和库存单独当成一个域(Domain)去对待是很必要的。

当然这个只是最初级的模型,接下来我会通过DDD中的一些核心概念的介绍,让大家更清楚的了解建模的过程。

领域事件(Domain Event)

An event is something that has happened in the past. A domain event is, logically, something that happened in a particular domain, and something you want other parts of the same domain (in-process) or domain in aonther bounded context to be aware of and potentially react to.

Domain Event是由一个特定领域触因为一个使用者Command触发的发生在过去的行为产生的事件,而这个事件是系统中其它部分感兴趣的。

为什么Domain Event如此重要? 因为在现在的分散式环境下,没有一个业务系统是割裂的,而Messaging绝对是系统之间耦合度最低,最健壮,最容易扩充套件的一种通讯机制。因此理论上它是分散式系统的必选项。

但是目前大部分系统的Event都设计的很随性,没有统一的指导和规范,导致Event滥用和无用的情况时有发生,而Domain Event给我们一个很好的方向,指引我们该如何设计我们系统的Event。

Event命名

Your Domain Event type names should be a statement of a past occurrence, that is, a verb in the past tense.

因为表示的是过去事件,所以推荐命名为Domain Name + 动词的过去式 + Event。这样比较可以确切的表达业务语义。

下面是几个举例:

1. CustomerCreatedEvent,表示客户建立后发出的领域事件。

2. OpportunityTransferedEvent,表示机会转移后发出的领域事件。

3. LeadsCreatedEvent,表示线索建立后发出的领域事件。

Event内容

Event的内容有两种形式:

(1)Enrichment:也就是在Event的payload中尽量多多放data,这样consumer就可以自恰(Autonomy)的处理讯息了。

(2)Query-Back:这种是在Event中通过回拨拿到更多的data,这种形式会加重系统的负载,performance也会差一些。

所以如果要在Enrichment和Query-Back之间做选择的话,首先推荐使用Enrichment。

Event Sourcing

Event Sourcing是在Domain Event上面的一个扩充套件,是一个可选项。也就是要有一个Event Store储存所有的Events,其实如果你是用MetaQ作为Event机制的话,这些Events都是储存在MetaQ当中的,只是MetaQ并没有提供很好的Event查询和回溯,所以如果决定使用Event Sourcing的话,最好还是自己单独建立一个Event Store。

使用Event Sourcing主要有以下好处,如果用不到的话,完全可以不用,但是Domain Event还是强烈建议要使用。

1、Event Sourcing储存了所有发生在Core Domain上面的事件。

2、基于这些事件,我们可以做系统回放,系统Debug,以及做使用者行为的分析(类似于打点)

Event Storming

事件风暴是《DDD Distilled》书中提出的一个业务分析的方法论,其主要作用是从Domian事件出发,来分析使用者Command,来找到Ubiquitous Languange,来抽象Domain Entity以及Bounded Context。

可以和User Story的方法论结合起来使用,其最大的优点是,这种分析方式即使是non-tech的人,比如产品,业务专家等也能听得懂,也能参与进来。相比较一上来就使用UML画领域模型图而言。

聚合根(Aggreagte)

聚合根(Aggregate Root)是DDD中的一个概念,是一种更大范围的封装,把一组有相同生命周期、在业务上不可分隔的实体和值物件放在一起考虑,只有根实体可以对外暴露引用,也是一种内聚性的表现。

确定聚合边界要满足固定规则(Invariant),是指在资料变化时必须保持的一致性规则,具体规则如下

- 根实体具有全域性标识,最终负责检查规定规则

- 聚合内的实体具有本地标识,这些标识在Aggregate内部才是唯一的

- 外部物件不能引用除根Entity之外的任何内部物件

- 只有Aggregate的根Entity才能直接通过数据库查询获取,其他物件必须通过遍历关联来发现

- Aggegate内部的物件可以保持对其他Aggregate根的引用

- Aggregate边界内的任何物件修改时,整个Aggregate的所有固定规则都必须满足

还是看银行的例子,Account(账号)是CustomerInfo(客户资讯)Entity和Address(值物件)的聚合根,Tansaction(交易)是流水(Journal)的聚合根,因为流水是因为交易才产生的,具有相同的生命周期。

最后提醒一下,聚合根是一个逻辑概念,主观性很强,所以在建模过程中很容易产生分歧,因此在日常工作中千万不要教条,把握住一条主要原则,我们的最终目的是为了业务语义显现化,如果因为聚合根把模型弄的晦涩难懂那就得不偿失了。

领域服务(Domain Service)

什么是领域服务

有些领域中的动作,它们是一些动词,看上去却不属于任何物件。它们代表了领域中的一个重要的行为,所以不能忽略它们或者简单地把它们合并到某个实体或者值物件中。当这样的行为从领域中被识别出来时,最佳实践是将它宣告成一个服务。这样的物件不再拥有内建的状态。它的作用仅仅是为领域提供相应的功能。

Service往往是以一个活动来命名,而不是Entity来命名。例如开篇转账的例子,转账(transfer)这个行为是一个非常重要的领域概念,但是它是发生在两个账号之间的,归属于账号Entity并不合适,因为一个账号Entity没有必要去关联他需要转账的账号Entity,这种情况下,使用MoneyTransferDomainService就比较合适了。

识别领域服务,主要看它是否满足以下三个特征:

1. 服务执行的操作代表了一个领域概念,这个领域概念无法自然地隶属于一个实体或者值物件。

2. 被执行的操作涉及到领域中的其他的物件。

3. 操作是无状态的。

领域服务陷阱

在使用领域服务时要特别当心,一个比较常见的错误是没有努力为行为找到一个适当的物件,就直接抽象成领域服务,这会使我们的程式码逐渐转化为过程式的程式设计,一个极端的例子是把所有的行为都放到领域服务中,而领域模型退化成只有属性的贫血DO,那DDD就没有任何意义了。

所以一定要深入思考,既不能勉强将行为放到不符合物件定义的物件中,破坏物件的内聚性,使其语义变得模糊。也不能不加思考的都放到领域服务中,从而退化成面向过程的程式设计。

应用服务和领域服务如何划分

在领域建模中,我们一般将系统划分三个大的层次,即应用层(Application Layer),领域层(Domain Layer)和基础实施层(Infrastructure Layer),关于这三个层次的详细内容可以参考我的另一篇SOFA框架的分层设计。

可以看到在App层和Domain层都有服务(Service),这两个Service如何划分呢,什么样的功能应该放在应用层,什么样的功能应该放在领域层呢?

决定一个服务(Service)应该归属于哪一层是很困难的。如果所执行的操作概念上属于应用层,那么服务就应该放到这个层。如果操作是关于领域物件的,而且确实是与领域有关的、为领域的需要服务,那么它就应该属于领域层。

总的来说,涉及到重要领域概念的行为应该放在Domain层,而其它非领域逻辑的技术程式码放在App层,例如引数的解析,上下文的组装,呼叫领域服务,讯息传送等。还是银行转账的case为例,下图给出了划分的建议:

边界上下文(Bounded Context)

领域实体是有边界上下文的,比如Apple这个实体不同的上下文,表达的含义就完全不一样,在水果店它就是水果,在苹果专卖店它就是手机。

所以边界上下文(Bounded Context)在DDD里面是一个非常重要的概念,Bounded Context明确地限定了模型的应用范围,在Context中,要保证模型在逻辑上统一,而不用考虑它是不是适用于边界之外的情况。在其他Context中,会使用其他模型,这些模型具有不同的术语、概念、规则和Ubiquitous Language的行话。

上下文对映(Context Mapping)

那么不同Context下的业务要互相通讯怎么办呢?这就涉及跨边界的Context Mapping(上下文对映),首先不同上下文之间的通讯可以是同步的,也可以是异步的,同步的话一般是RPC或者RESTFul,异步的话会推荐上文提到的Domain Event.

Mapping的方式有很多种,有Shared Kernal(共享核心),Conformist(追随者),以及Anti-Corruption(防腐层)等等。

我个人比较推崇Domain Event + AC,这样可以将系统之间的耦合降到最低。

以我们真实的业务场景举个例子,比如会员这个概念在ICBU网站是指网站上的Buyer,但是在CRM领域是指Customer,虽然很多的属性都是一样的,但是二者在不同的Context下其语义和概念是有差别的,我们需要用AC做一下转换:

边界上下文和微服务

先来说一下微服务,抛开以Docker为代表的底层容器化技术不看,微服务和我们之前的SOA么有本质区别。

这不是我一个人的观点,关于这个想法我还专门求证了业界大牛Randy Shoup,问他微服务和SOA的区别,下面是他给我的回答

那么如何划分系统,才能得到一个比较合适的粒度,不会太粗,也不会太细呢。此时我们可以考虑DDD的战略设计,即从战略角度整体描述业务领域全貌,然后通过边界上下文将不同的实体归类到相对应的域里面。

比如在CRM领域,我们按照下面的战略设计图,我会自然的把CRM系统划分成销售服务,组织许可权服务,营销服务,售卖服务。

模型重构

最后我想强调的是,建模不是一次性的工作,也不可能是一次性的工作,业务在演化,随之而来的模型也需要演化和重构,当模型和业务不匹配的时候,你还是要霸王硬上弓的往里面塞,其结果可想而知。

模型统一

建模的过程很像盲人摸象,不同背景人用不同的视角看同一个东西,其理解也是不一样的。比如两个盲人都摸到大象鼻子,一个人认为是像蛇(活的能动),而另一个人认为像消防水管(可以喷水),那么他们将很难整合。

双方都无法接受对方的模型,因为那不符合自己的体验。事实上,他们需要一个新的抽象,这个抽象需要把蛇的“活着的特性”与消防水管的“喷水功能”合并到一起,而这个抽象还应该排除先前两个模型中一些不确切的含义和属性,比如毒牙,或者卷起来放到消防车上去的行为,这就是模型的统一。

统一完的模型也许还不叫大象鼻子,但是已经很接近大象鼻子的属性和功能了,随着我们对模型物件、对业务理解的越来越深入、越来越透彻,我们会不断的调整演化我们的模型,所以建模不是一个one-time off的工作,而是一个持续不断演化重构的过程。

模型演化

世界上唯一不变的就是变化,模型和程式码一样也需要不断的重构和精化,每一次的精化之后,开发人员应该对领域知识有了更加清晰的认识。这使得理解上的突破成为可能,之后,一系列快速的改变得到了更符合使用者需要并更加切合实际的模型。其功能性及说明性急速增强,而复杂性却随之消失。

这种突破需要我们对业务有更加深刻的领悟和思考,然后再加上重构的勇气和能力,勇气是专案工期很紧你敢不敢重构,能力是你有没有完备的CI保证你的重构不破坏现有的业务逻辑。

实体在演变

以开篇的银行账号为例,假如一开始账号都有withdraw(取钱)的行为,此时只需要在Account上加上withdraw方法就好了。

- 演变一:

随着业务的发展,我们需要支援ATM账号和Online账号,而Online账号是不能withdraw的,此时最差的做法是保持模型不变,而是在withdraw方法中判断如果是OnlineAccount则丢掷异常。这种简单的程式码堆砌可以满足业务功能,但是其业务语义完全被掩盖。更好的重构方法应该是将withdraw抽成一个界面IWithdrawable。

- 演变二:

好的,没有什么可以阻挡业务对变化的向往。现在公司出于安全性的考虑,为新开通的ATMAccount设定了取款上线,超过则不能支取。简单做法是在IWithdrawable中再加入一个setLimit行为,可是我们并不想改动影响到老的账号业务,所以更好的重构应该是重新写一个ILimitedWithdrawable界面,让其继承老界面,这样老的程式码就可以保持不变了。

通过上面的例子,我们可以看到领域模型和面向物件是一对孪生兄弟,我们会用到大量的OO原则,比如上面的重构就用到了SOLID的SRP(单一职责)和OCP(开闭原则)。在实际工作中,我的确也有这样的体会,自从践行DDD以后,我们采用OOA和OOD的时候比以前明显多了很多,OO的能力也在不断的提升。

引入新抽象

还是以开篇的转账来举个例子,假如转账业务开始变的复杂,要支援现金,信用卡,支付宝,比特币等多种通道,且没种通道的约束不一样,还要支援一对多的转账。那么你还是用一个transfer(fromAccount, toAccount)就不合适了,可能需要抽象出一个专门的领域物件Transaction,这样才能更好的表达业务,其演化过程如下:

业务视觉化和可配置化

好的领域建模可以降低应用的复杂性,而视觉化和可配置化主要是帮助大家(主要是非技术人员,比如产品,业务和客户)直观地了解系统和配置系统,提供了一种“code free”的解决方案,也是SaaS软件的主要卖点。

要注意的是视觉化和可配置化难免会给系统增加额外的复杂度,必须慎之又慎,最好是能使视觉化和配置化的逻辑与业务逻辑尽量少的耦合,否则破坏了原有的架构,把事情搞的更复杂就得不偿失了。

在可扩充套件设计中,我已经介绍了我们SOFA架构是如何通过扩充套件点的设计来支撑不同业务差异化的需求的,那么可否更进一步,我们将领域的行为(也叫能力)和扩充套件点用视觉化的方式呈现出来,并对于一些不需要编码实现的扩充套件点用配置的方式去完成呢。

当然是可以的,比如还是开篇转账的例子,对于透支策略OverdraftPolicy这个业务扩充套件点,新来一个业务说透支额度不能超过1000,我们可以完全结合规则引擎进行配置化完成,而不需要编码。

所以我能想到的一种还比较优雅的方式,是通过Annotation注解的方式对领域能力和扩充套件点进行标注,然后在系统bootstrap阶段,通过程式码扫描的方式,将这些能力点和扩充套件点收集起来上传到中心服务器,然后再通过GUI的方式呈现出来,从而做到业务的视觉化和可配置化。大概的示意图如下:

有同学可能会问流程要不要视觉化,这里要分清楚两个概念,业务逻辑流和工作流,很多同学混淆了这两个概念。

业务逻辑流是响应一次使用者请求的业务处理过程,其本身就是业务逻辑,对其编排和视觉化的意义并不是很大,无外乎只是把程式码逻辑可视化了,在我们的SOFA框架中,是通过扩充套件点和策略模式来处理业务的分支情况,而我看到我们阿里很多的内部系统将这种响应一次使用者请求的业务逻辑用很重的工作流引擎来做,美其名曰流程可编排,实质上往往是把简单的事情复杂化了。

而工作流是指完成一项任务所需要不同节点的连线,节点主要分为自动节点和人工节点,其中每个人工节点都需要使用者的参与,也就是响应一次使用者的请求,比如审批流程中的经理审批节点,CRM销售过程的业务员的处理节点等等。

此时可以考虑使用工作流引擎,特别是当你的系统需要让使用者自定义流程的时候,那就不得不使用视觉化和可配置的工作流引擎了,除此之外,最好不要自找麻烦。

我曾在银行工作过,亲眼看见过IBM是怎么忽悠银行使用它们的BPM系统,然后把系统弄的巨复杂无比,所以我对工作流引擎的印象并不好,当然也不排除有用的特别合适的案例,只是我还没看见,如果有看见的同学麻烦告诉我一声,学习一下。

因为我们现在还没有让使用者自定义流程的诉求,所以使用工作流引擎并不在我们现阶段的考虑范围之内。

您的转发+关注就是对笔者最大的支援,欢迎关注。

对大厂架构设计,BAT等厂家面试题解读,程式语言理论或者互联网圈逸闻趣事这些感兴趣,欢迎关注笔者,没有错,干货文章都在这里。
2019-07-04 14:49:00

相关文章