Spark中master与worker的进程RPC通信实现
1.构建master的actor
package SparkRPC
import akka.actor.{Actor, ActorSystem, Props}
import com.typesafe.config.ConfigFactory
import scala.collection.mutable
/**
* Created by hqs on 2018/1/24.
* 1.启动master,启动worker
* 2.worker启动后连接master,发送注册消息(封装起来)
* 3.master受到注册消息并保存,返回注册成功消息给worker
* 4.worker启动一个定时任务,发送心跳给master(先发送给自己,在发送给master)
* 5.master接收心跳消息,更新保存的心跳信息
*
* 6.master主动启动一个定时任务,检查心跳时间是否超过设定值,若超过,则删除worker的注册信息。
*/
class Master extends Actor {
private val workerMp: mutable.HashMap[String, WorkerInfo] = new mutable.HashMap[String, WorkerInfo]()
override def preStart(): Unit = {
//启动定时任务去检查是否有死去的worker
import scala.concurrent.duration._
import context.dispatcher
context.system.scheduler.schedule(0 second, 15 second, self, CheckWorker)
}
override def receive: Receive = {
case "start" => println("master start...")
//接收并注册,返回成功消息。
case Register2Master(workerId, cores, memory) => {
workerMp(workerId) = new WorkerInfo(cores, memory)
println(s"add a worker,workerId = ${workerId}")
println(s"now total workers = ${workerMp.size}")
sender() ! RegisSuccess
}
//接收心跳,更新信息
case HeartBeat(workerId) => {
if (workerMp.contains(workerId)) {
workerMp(workerId).lastloginTime = System.currentTimeMillis()
}
}
case CheckWorker => {
//过滤出已经超时的worker,大于两个心跳认为超时。
val deadWorkers = workerMp.filter({
mp => {
System.currentTimeMillis() - mp._2.lastloginTime > 20 * 1000
}
})
//用一个map来减去另外一个map
workerMp --= deadWorkers.map(mp => mp._1)
println(s"now total workers = ${workerMp.size}")
}
}
}
object Master {
val MASTER_ACS_NAME = "master_acs_name"
val MASTER_AC_NAME = "master_ac_name"
def main(args: Array[String]): Unit = {
if (args.length != 2) {
println("Master <masterIp,masterPort>")
sys.exit()
}
val Array(masterIp, masterPort) = args
val str =
s"""
|akka.actor.provider = "akka.remote.RemoteActorRefProvider"
|akka.remote.netty.tcp.hostname = "${masterIp}"
|akka.remote.netty.tcp.port = "${masterPort}"
""".stripMargin
val conf = ConfigFactory.parseString(str)
val acs: ActorSystem = ActorSystem.create(MASTER_ACS_NAME, conf)
val masterRef = acs.actorOf(Props(new Master), MASTER_AC_NAME)
masterRef ! "start"
}
}
2.构建worker的actor
package SparkRPC
import java.util.UUID
import akka.actor.{Actor, ActorSelection, ActorSystem, Props}
import com.typesafe.config.ConfigFactory
/**
* Created by hqs on 2018/1/24.
*/
class Worker(val masterIp:String,val masterPort:Int,val cores:Int,val memory:Int) extends Actor{
val workerId = UUID.randomUUID().toString
var masSele: ActorSelection = null
//注册worker信息
override def preStart(): Unit = {
//取得master的路径
val path = s"akka.tcp://${Master.MASTER_ACS_NAME}@${masterIp}:${masterPort}/user/${Master.MASTER_AC_NAME}"
masSele = context.actorSelection(path)
masSele ! Register2Master(workerId,cores,memory)
}
override def receive: Receive = {
case "start" => println("worker starting")
//发送定时心跳信息
case RegisSuccess => {
println("success start scheduler")
/**
* initialDelay: FiniteDuration, 延迟时间 延迟启动定时任务的时间
* interval: FiniteDuration, 间隔时间 每隔多长时间
* receiver: ActorRef, 信息发给谁 接收方
* message: Any 发送的信息 封装成case class
*/
//导入时间单位,启动定时任务。
import scala.concurrent.duration._
import context.dispatcher
context.system.scheduler.schedule(0 second,10 second,self,SendHeartBeat)
}
case SendHeartBeat => {
masSele ! HeartBeat(workerId)
println("worker 向 master 发送心跳信息...")
}
}
}
object Worker{
val WORKER_ACS_NAME = "worker_acs_name"
val WORKER_AC_NAME = "worker_ac_name"
def main(args: Array[String]): Unit = {
if(args.length != 6){
println("Worker <masterIp,masterPort,workerIp,workerPort,cores,memory>")
sys.exit()
}
val Array(masterIp,masterPort,workerIp,workerPort,cores,memory) = args
val str =
s"""
|akka.actor.provider = "akka.remote.RemoteActorRefProvider"
|akka.remote.netty.tcp.hostname = "${workerIp}"
|akka.remote.netty.tcp.port = "${workerPort}"
""".stripMargin
val conf = ConfigFactory.parseString(str)
val acs = ActorSystem.create(WORKER_ACS_NAME,conf)
val scRef = acs.actorOf(Props(new Worker(masterIp,masterPort.toInt,cores.toInt,memory.toInt)),WORKER_AC_NAME)
scRef ! "start"
}
}
3.master与worker的消息传递封装
package SparkRPC
/**
* Created by hqs on 2018/1/27.
*/
class Message {
}
//worker发送注册消息
case class Register2Master(workerId:String,cores:Int,memory:Int)
//master返回注册成功的消息
case object RegisSuccess
//发送心跳给自己
case object SendHeartBeat
//发送心跳给master
case class HeartBeat(workerId:String)
//master定时检查worker存活状态
case object CheckWorker
4.总结:master与worker依赖于akka的actor来实现通信。会产生定时心跳任务,检查超时的worker。
Spark中master与worker的进程RPC通信实现的更多相关文章
- 【原】Spark中Master源码分析(二)
继续上一篇的内容.上一篇的内容为: Spark中Master源码分析(一) http://www.cnblogs.com/yourarebest/p/5312965.html 4.receive方法, ...
- <spark> error:启动spark后查看进程,进程中master和worker进程冲突
启动hadoop再启动spark后jps,发现master进程和worker进程同时存在,调试了半天配置文件. 测试发现,当我关闭hadoop后 worker进程还是存在, 但是,当我再关闭spark ...
- 【原】Spark中Master源码分析(一)
Master作为集群的Manager,对于集群的健壮运行发挥着十分重要的作用.下面,我们一起了解一下Master是听从Client(Leader)的号召,如何管理好Worker的吧. 1.家当(静态属 ...
- Spark技术内幕:Client,Master和Worker 通信源码解析
http://blog.csdn.net/anzhsoft/article/details/30802603 Spark的Cluster Manager可以有几种部署模式: Standlone Mes ...
- Spark技术内幕:Client,Master和Worker 通信源代码解析
Spark的Cluster Manager能够有几种部署模式: Standlone Mesos YARN EC2 Local 在向集群提交计算任务后,系统的运算模型就是Driver Program定义 ...
- 【原】 Spark中Worker源码分析(一)
Worker作为对于Spark集群的健壮运行起着举足轻重的作用,作为Master的奴隶,每15s向Master告诉自己还活着,一旦主人(Master>有了任务(Application),立马交给 ...
- 【原】 Spark中Worker源码分析(二)
继续前一篇的内容.前一篇内容为: Spark中Worker源码分析(一)http://www.cnblogs.com/yourarebest/p/5300202.html 4.receive方法, r ...
- Spark分析之Master、Worker以及Application三者之间如何建立连接
Master.preStart(){ webUi.bind() context.system.scheduler.schedule( millis, WORKER_TIMEOUT millis, se ...
- Spark 中的 RPC 的几个类
Spark 中 RPC 部分的涉及了几个类,有点晕,在此记录一下 1. RpcEndpoint: RPC的一个端点.给定了相应消息的触发函数.保证 `onStart`, `receive` and ...
随机推荐
- 从开启GTID功能的库同步数据到未开启GTID功能库时,注意事项!
从开启GTID的库中导出数据到未开启GTID的库中,需要注意,在导出的文件中去掉相应的gtid内容,否则导入时会报错如下: ERROR 1839 (HY000) at line 24 in file: ...
- Java中资料的上传与下载
1.导架包 <dependency> <groupId>commons-fileupload</groupId> <artifactId>commons ...
- Android中的EventBus
1.分析 EventBus是一个针对Android的事件发布和订阅的框架,主要功能是替代Intent,Handler,BroadCast在Fragment,Activity,Service,线程之间传 ...
- 微信公众号UX分析—— 学生作业小结
1. 不足: 1. 权威性:个人帐号,显得不够正式. 2. 排版问题: + 没有必要的外接端口,界面设计极度缺少排版.哪怕是个人公众号都不至于如此,更何况这是一个学校的教务平台. 3. 反应不及时或无 ...
- 如果你要查看文件的每个部分是谁修改的, 那么 git blame 就是不二选择
原文: http://gitbook.liuhui998.com/5_5.html 如果你要查看文件的每个部分是谁修改的, 那么 git blame 就是不二选择. 只要运行'git blame [f ...
- php安装扩展的几种方法
转自:http://doc3.workerman.net/appendices/install-extension.html 安装扩展 注意 与Apache+PHP或者Nginx+PHP的运行模式不同 ...
- Android Service用法知识点的讲解
Android Service 学习Service相关知识点: android service 的基础知识,生命周期,service分类,运行地点(本地服务,远程服务),运行类型(前台服务,后台服务) ...
- Linux - 基础命令汇总
珠玉在前,不再赘言. 60个命令 对Linux新手非常有用的20个命令 对中级Linux用户非常有用的20个命令 对Linux专家非常有用的20个命令 一些技巧 最实用的 Linux 命令行使用技巧 ...
- 关于Ubuntu拒绝root用户ssh远程登录
#sudo vim /etc/ssh/sshd_config 找到并用#注释掉这行:PermitRootLogin prohibit-password 新建一行 添加:PermitRootLogin ...
- HDU 4570---Multi-bit Trie(区间DP)
题目链接 Problem Description IP lookup is one of the key functions of routers for packets forwarding and ...