在之前的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. Java常见问题汇总

    1.String,StringBuffer,StringBulider的区别及应用场景 2.Servlet生命周期 3.向上转型与向下转型 4.Java的多态性 5.重写和重载的区别 6.深拷贝和浅拷 ...

  2. vmware安装kvm虚拟机

    1. 概述 本篇博客主要使用运行在win10专业版上的vmware workstation 15 pro虚拟化软件,安装centos7.7最小化系统,并在centos7上安装kvm虚拟机,实现快速创建 ...

  3. context.startActivity(Intent intent)方法启动activity

    在一个Activity环境中用该方法启动一个一个activity不会出任何问题,但在activity之外的其他组件中使用该方法就会出现以下错误: Calling startActivity() fro ...

  4. Simscape Multibody 教程 —— 入门学习

    写在前面 本文要点: Simscape Multibody 简介 Simscape Multibody 入门学习的推荐学习材料和学习顺序 建模仿真过程中的重要知识 模型的参数设置(Model Work ...

  5. 【搞定面试官】- Synchronized如何实现同步?锁优化?(1)

    前言 说起Java面试中最高频的知识点非多线程莫属.每每提起多线程都绕不过一个Java关键字--synchronized.我们都知道该关键字可以保证在同一时刻,只有一个线程可以执行某个方法或者某个代码 ...

  6. 2019icpc徐州现场赛 H Yuuki and a problem (树状数组套主席树)

    题意 2e5的数组,q个操作 1.将\(a[x]\)改为y 2.求下标l到r内所有的\(a[i]\)通过加法不能构成的最小的值 思路 通过二操作可以知道需要提取l到r内的值及其数量,而提取下标为l到r ...

  7. tensorboard的简单使用

    1. 首先保证你已有程序,下面是MLP实现手写数字分类模型的代码实现. 不懂的可以对照注释理解. #输入数据是28*28大小的图片,输出为10个类别,隐层大小为300个节点 from tensorfl ...

  8. Qt 条件编译 arm windows linux 判断 跨平台

    如果代码里面有些判断需要不同的参数做判断: 办法:在pro文件里面做定义 方法1:直接定义一个宏:用的时候可以直接判断,这样做不好的地方是编译前需要重新切换一下宏 1)定义宏 DEFINES += _ ...

  9. linux 删除文件 磁盘空间未释放

    具体情况就是:删除了一个超大文件后,发现磁盘空间没有变化 原因:有进程正在使用这个文件,虽然我们从文件系统的目录结构上解除链接(unlink),然而文件是被 打开的(有一个进程正在使用),那么进程将仍 ...

  10. 软件bug描述(android)

    1.bug主题:主要操作+bug主题 主题要简单明了,即开发一看主题就知道该问题. 2.描述: 作用:便于开发重现和定位缺陷的 2.1前置条件 2.2操作步骤 2.3预期结果 2.4实际结果 2.5备 ...