分散与聚合:简单说就是一个任务须要拆分红多个小任务,每一个小任务执行完后再把结果聚合在一块儿返回。git
代码 http://git.oschina.net/yangbajing/akka-actiongithub
本实例来自一个真实的线上产品,现将其需求简化以下:dom
key
,根据key
从网上抓取相关新闻duration
,设置任务到期时必须反回数据(返回实际已抓取数据)根据需求,一个简化的分散、聚合模式能够使用两个actor
来实现。ide
NewsTask测试
override def metricPreStart(): Unit = { context.system.scheduler.scheduleOnce(doneDuration, self, TaskDelay) } override def metricReceive: Receive = { case StartFetchNews => _receipt = sender() (0 until NewsTask.TASK_SIZE).foreach { i => context.actorOf(SearchPageTask.props(self), "scatter-" + i) ! SearchPage(key) } case GetNewsItem(newsItem) => _newses ::= newsItem if (_newses.size == NewsTask.TASK_SIZE) { logger.debug(s"分散任务,${NewsTask.TASK_SIZE}个已所有完成") if (_receipt != null) { _receipt ! NewsResult(key, _newses) _receipt = null } self ! PoisonPill } case TaskDelay => if (_receipt != null) { _receipt ! NewsResult(key, _newses) _receipt = null } }
metricPreStart
方法中设置定时方法,调用时间为从代码运行开始到doneDuration
时间为止。定时被触发时将向当前Actor
发送一个TaskDelay
消息。scala
在metricReceive
方法中,分别对StartFetchNews
、GetNewsItem
、TaskDelay
三个消息进行操做。debug
在收到StartFetchNews
消息时,actor首先保存发送者actor的引用(结果将返回到此actor)。再根据TASK_SIZE
生成相应子任务设计
GetNewsItem
消息的处理中,每收到一个消息就将其添加到_newses
列表中。并判断当_newses
个数等于TASK_SIZE
时(全部子任务已完成)将结果发送给_receipt
。code
self ! PoisonPill
,这句代码中止actor自身。它将把“毒药”发送到NewsTask
Actor的接收邮箱队列中。
TaskDelay
消息被触发时,将直接返回已完成的新闻_newses
。返回数据后并不终止当前还未运行完任务。
SearchPageTask
override def metricReceive: Receive = { case SearchPage(key) => // XXX 模拟抓取新闻时间 TimeUtils.sleep(Random.nextInt(20).seconds) val item = NewsItem( "http://newssite/news/" + self.path.name, "测试新闻" + self.path.name, self.path.name, TimeUtils.now().toString, "内容简介", "新闻正文") taskRef ! GetNewsItem(item) context.stop(self) }
SearchPageTask
的代码逻辑就比较易懂了,这里使用sleep
来模拟实际抓取新闻时的耗时。生成结果后返回数据给`taskRef`,并终止本身。
./sbt akka-action > testOnly me.yangbajing.akkaaction.scattergather.ScatterGatherTest
这是一个简单的Akka
实例,实现了任务分发与结果聚合。提供了一种在指定时间内返回部份有效数据,同时任务继续执行的方式。这种分散、聚合的模式在实际生产中很经常使用,好比对多种数据源的整合,或某些须要长时间运行同时对返回数据完整性无强制要求的状况等。
MetricActor
演示了怎么自定义Actor
,并为其提供一些侦测点的方式。之后有时间会写篇详文介绍。