在之前的akka设计模式系列-消息模型中,我们介绍了akka的消息设计方案,但随着实践的深入,发现了一些问题,这里重新梳理一下设计方法,避免之前的错误、不当的观点给大家带来误解。


命令和事件

  我们仍然把akka中的消息分为命令和事件两大类,但二者的具体含义和实现有一点变化。“命令,是指一个actor给另外一个actor发送指令做相关的业务逻辑;事件,则是actor对某个命令的响应结果,或者对其他事件的响应结果”。之前是这样定义的,但在具体的实践过程中发现了一些问题。

  比如,命令如何归类?是根据命令的发送者归类还是根据接收者归类呢?是不是根据消息的作用对象更恰当呢?事件的定义是不是可以更加宽泛一下呢?如果某个消息是定时产生的,那它应该是一个命令还是一个消息呢?

  当然还有一些其他的问题,认真总结后,在代码设计上我们做了一些优化。

消息

trait Message {
/**
* 消息发生的时间
*/
final lazy val at: Long = System.currentTimeMillis()
}

  消息的定义很简单,只有一个时间字段,表示命令或事件的产生时间。但在实践过程中发现,很少使用这个字段,再加上System.currentTimeMillis()多线程调用会有性能问题,所以干脆把它定义成了lazy。关于这一点,大家可以根据各自的实际情况修改。

命令

trait Reply{
def replyTo:ActorRef
} trait Command extends Message with Reply

  命令就是含有回复对象的消息,之所以强化这一点,主要是想 用Command模拟能返回结果的函数。也就是说,actor收到Command之后必须有一个返回事件。

trait TaskCommand extends Command
object TaskCommand{
final case class Run(jobContext: JobContext,replyTo:ActorRef) extends TaskCommand
final case class Kill(jobId:UID, taskId:UID, executorType: ExecutorType, replyTo:ActorRef) extends TaskCommand
}

  上面定义两个命令:Run、Kill。很明显我们是根据命令的作用对象进行归类,也就是实现统一的trait(TaskCommand)。之所以这样做是因为我们发现很多命令被某一个actor处理之后,并不返回结果,而是发送给下一个actor处理,并由其返回结果。这种情况下无论从发送者还是接收者归类都不恰当,而用命令的作用对象进行归类则比较合适。

事件

trait SchedulerEvent extends Event
object SchedulerEvent{
final case class ScheduleTriggered() extends SchedulerEvent
final case class NodeDowned(nodeAnchor:String) extends SchedulerEvent
}

   事件的含义和实现方法跟之前基本不变,但总体来说也是围绕某个对象进行归类。比如Scheduler这个对象有两个事件:ScheduleTriggered、NodeDowned。这样事件在多个actor传递的时候不会改变其原始的含义,比较容易理解。

命令结果

  actor处理命令的时候,需要有一个返回结果,仍然需要对如何返回进行界定。我们根据面向过程编程的一个优良习惯进行规范:对某一类Command返回结果时,只能有一个地方返回成功结果,可以有多个地方返回失败结果。这只是一个软性约束,但非常有用,这样排查问题比较方便,也有利于画系统的流程图。

总结

  在Akka中消息设计的好坏往往决定着应用的质量,非常重要。每个人的设计理念不同,实现的方式也就不同,这里也只是抛砖引玉,如果你有更好的设计模式,欢迎留言或者加微信(HelloGrape)讨论。

akka设计模式系列-消息模型(续)的更多相关文章

  1. akka设计模式系列-消息模型

    通过前面的文章我们总结了几个常见的actor设计模式,但此处不得不提前介绍一下在Akka中消息的设计模式.随着对Akka的使用,我们会发现,使用Akka设计系统其实就是面向消息编程.actor之间消息 ...

  2. akka设计模式系列-慎用ask

    慎用ask应该是Akka设计的一个准则,很多时候我们应该禁用ask.之所以单独把ask拎出来作为一篇博文,主要是akka的初学者往往对ask的使用比较疑惑. "Using ask will ...

  3. akka设计模式系列

    由于本人爱好Scala,顺便也就爱好Akka,但目前网上对Akka的介绍大多都是概念上或技术方向上的介绍,基本没有Akka设计模式或者Actor模型设计模式的资料.这对于Akka的普及非常不利,因为即 ...

  4. akka设计模式系列(Actor模型)

    谈到Akka就必须介绍Actor并发模型,而谈到Actor就必须看一篇叫做<A Universal Modular Actor Formalism for Artificial Intellig ...

  5. akka设计模式系列-While模式

    While模式严格来说是while循环在Akka中的合理实现.while是开发过程中经常用到的语句之一,也是绝大部分编程语言都支持的语法.但while语句是一个循环,如果循环条件没有达到会一直执行wh ...

  6. akka设计模式系列-Backend模式

    上一节我们介绍了Akka使用的基本模式,简单点来说就是,发消息给actor,处理结束后返回消息.但这种模式有个缺陷,就是一旦某个消息处理的比较慢,就会阻塞后面所有消息的处理.那么有没有方法规避这种阻塞 ...

  7. akka设计模式系列-actor锚定

    actor锚定模式是指使用actorSelection对acor进行锚定的设计模式,也可以说是一个对actor的引用技巧.在某些情况下,我们可能需要能够根据Actor的path锚定对应的实例.简单来说 ...

  8. akka设计模式系列-Chain模式

    链式调用在很多框架和系统中经常存在,算不得上是我自己总结的设计模式,此处只是简单介绍在Akka中的两种实现方式.我在这边博客中简化了链式调用的场景,简化后也更符合Akka的设计哲学. trait Ch ...

  9. akka设计模式系列-akka在秒杀场景的应用

    本博客讨论一下akka在秒杀场景下的应用,提出自己的见解,只做抛砖引玉,大神勿喷.秒杀活动涉及到前中后台各个阶段,为了说明问题,我们简化场景,只研究akka在后台如何处理秒杀业务. 秒杀活动 所谓的秒 ...

随机推荐

  1. docker扫盲,面试连这都不会就等着挂吧!

    现在很多公司项目部署都是采用K8S docker容器方式,出门面试被问的概率极大,如果被面试官问docker相关知识点直接懵逼,那么基本就是被pass了,除非其他方面技术过硬.所以这种相对前沿的技术, ...

  2. ElasticSearch系列专栏

    最近我们公司因业务发展较快,不少服务遇到了一些瓶颈,影响最大的就是数据量的暴增带来的搜索效率的问题.虽然建立索引以及利用好缓存可以有效地缓解该问题,但是随着业务的发展,业务的复杂度也逐渐提升,原有的技 ...

  3. 深入了解Zookeeper

    ZooKeeper是一个分布式的,开放源码的分布式应用程序协调服务,它包含一个简单的原语集,分布式应用程序可以基于它实现同步服务,配置维护和命名服务等.Zookeeper是hadoop的一个子项目,其 ...

  4. AcWing 789.数的范围

    AcWing 789.数的范围 题目描述 给定一个按照升序排列的长度为n的整数数组,以及 q 个查询. 对于每个查询,返回一个元素k的起始位置和终止位置(位置从0开始计数). 如果数组中不存在该元素, ...

  5. 【读书笔记】关于《精通C#(第6版)》与《C#5.0图解教程》中的一点矛盾的地方

    志铭-2020年2月8日 03:32:03 先说明,这是一个旧问题,很久很久以前大家就讨论了, 哈哈哈,而且先声明这是一个很无聊的问题,

  6. 在VMware中如何清理多余的空间

    问题描述 平时用的编程计算机只有250G空间,c盘和d盘,今天准备做实验,发现删除虚拟机中系统的内容不但没有减少空间,反而增加了,这时我意识到虚拟机内部可能与咱们想象的操作模式不一样. 解决办法 我的 ...

  7. Codeforces_825

    A.连续1的个数,0用来分割,注意连续的0. #include<bits/stdc++.h> using namespace std; int n; string s; int main( ...

  8. 在家想自学Java,有C语言底子,请问哪本书适合?

    一.问题剖析 看到这个问题,我想吹水两句再做推荐.一般发出这个疑问都处在初学编程阶段,编程语言都是相通的,只要你领悟了一门语言的"任督二脉",以后你学哪一门语言都会轻易上手.学语言 ...

  9. Go语言实现:【剑指offer】变态跳台阶

    该题目来源于牛客网<剑指offer>专题. 一只青蛙一次可以跳上1级台阶,也可以跳上2级--它也可以跳上n级.求该青蛙跳上一个n级的台阶总共有多少种跳法. 找规律: 1阶:1种: 2阶:2 ...

  10. c++ 内存分配中一个有趣的小问题

    以下代码测试环境:vs2019 执行这么一段代码,看看会发生什么. int main() { ] = { }; arr[] = ; } 毫无疑问,会报错,因为访问越界了. 再看看另一段代码 ] = { ...