在之前的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. 缓存 ehcache

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

  2. laravel 初学路由简单介绍

    文档中的路由详细演示[初学laravel]对应laravel 的框架目录:routes/web.php 路由的格式一:Route::get($uri,$callback); 1.简单的浏览器输出 Ro ...

  3. 看片微信号+薇myy9199买片微信号+myy9199绝对靠谱号

    最新看片卖片微信号+myy9199,2020最新有效靠谱号,诚信有效,死链包换,2019年11月,我决定学习计算机编程,以java语言为主.我就读于传统工科专业,没怎么接触过计算机相关概念与课程,我知 ...

  4. 配置微软Azure大数据HDInsight云集群

    配置微软Azure大数据HDInsight云集群,存储账户.托管标识等问题也都参考官方文档解决了. 原文在我的开源中国博客 https://my.oschina.net/finchxu/blog/31 ...

  5. 二、Linux系统硬链接和软链接详细介绍与实践

    链接的概念 在linux系统中,链接可分为两种:一种被称为硬链接(Hard LinK),另一种被称为软链接或符号链接(Symbolic Link).在默认不带参数的情况下,执行ln命令创建的链接是硬链 ...

  6. tomcat的编码设置

    Connector port="8080" protocol="HTTP/1.1" URIEncoding="UTF-8"          ...

  7. java与c++,python

    Java与C++的异同点总结 C++/C/JAVA/Python之间的区别? C++语言与Java语言的区别有哪些? java与C++的区别

  8. HDU_1166_树状数组

    http://acm.hdu.edu.cn/showproblem.php?pid=1166 树状数组入门题. #include<iostream> #include<cstring ...

  9. linux入门系列12--磁盘管理之分区、格式化与挂载

    前面系列文章讲解了VI编辑器.常用命令.防火墙及网络服务管理,本篇将讲解磁盘管理相关知识. 本文将会介绍大量的Linux命令,其中有一部分在"linux入门系列5--新手必会的linux命令 ...

  10. Python socket 基础(Client) - Foundations of Python Socket

    Python socket 基础- Foundations of Python Socket 建立socket - establish socket import socket s = socket. ...