在之前的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. Centos与Ubuntu

    共同点 1.两个系统都分别有桌面系统与服务器系统,不过ubuntu的桌面从外观上来看要比centos的漂亮 不同点 1.centos中新建的普通用户是没有sudo权限的,如果想让普通用户拥有sudo权 ...

  2. 缓存 ehcache

    只是用于自己记录,防止日后忘记,回看所用 第一步:配置ehcahe 缓存 <?xml version="1.0" encoding="UTF-8"?> ...

  3. 最新2.7版本丨DataPipeline数据融合产品最新版本发布

    此次发布的2.7版本在进一步优化产品底层数据处理逻辑的同时更加注重提升用户在数据融合任务的日常管理.运行监控及资源分配等管理方面的功能增强与优化,力求帮助大家更为直观.便捷.稳定地管理数据融合任务,提 ...

  4. C++输出中文字符

    注:本文转载自互联网,感谢作者整理!   1. cout 场景1: 在源文件中定义 const char* str = "中文" 在 VC++ 编译器上,由于Windows环境用 ...

  5. Huffman编码和解码

    一.Huffman树 定义: 给定n个权值作为n个叶子结点,构造一棵二叉树,若该树的带权路径达到最小,这样的二叉树称为最优二叉树,也称为霍夫曼树(Huffman树). 特点:     Huffman树 ...

  6. 自己动手写个异步IO函数 --(基于 c# Task)

    前言    对于服务端,达到高性能.高扩展离不开异步.对于客户端,函数执行时间是1毫秒还是100毫秒差别不大,没必要为这一点点时间煞费苦心.对于异步,好多人还有误解,如: 异步就是多线程:异步就是如何 ...

  7. python学习记录(九)

    0911--https://www.cnblogs.com/fnng/archive/2013/05/08/3066054.html 魔法方法.属性 准备工作 为了确保是新型类,应该把_metacla ...

  8. android studio sqlite实际应用中存在的问题

    原项目已上传到github long f = dbdatabase.update("user", values, "id=?", new String[]{St ...

  9. .NET异步程序设计之任务并行库

    目录 1.简介 2.Parallel类 2.0 Parallel类简介 2.1 Parallel.For() 2.2 Parallel.ForEach() 2.3 Parallel.Invoke() ...

  10. 第3章 JDK并发包(四)

    3.2.5 自定义线程创建:ThreadFactory 线程池的主要作用是为了线程复用,也就是避免了线程的频繁创建. ThreadFactory是一个接口,它只有一个方法,用来创建线程: Thread ...