ZooKeeper - 攘外安内的多面管理员
Zookeeper 作为分布式协调框架,在实际生产中有着不可替代的位置,它作为管理框架,大多数时间是不需要我们去维护的。也正是由于它是协调全局的管理者,所以它的一些内部原理我们还是要了解的。
今天我们就从以下几个方面,来深入探讨下 ZK 的工作原理:
攘外 :Zookeeper 的读写过程
安内 : 集群 leader 选举方式
图解攘外安内背后的 ZAB 算法
ZK 状态同步
ZK 分布式锁的应用
01 攘外: 图说 ZK 的读写过程
写过程:
1 客户端向 zk 集群写入数据,如 create /dsjjh ;与一个 follower 建立 session 连接,连接的是 从节点 follower 01
2 从节点 follower 01 将写请求转发给 leader
3 leader收到消息发出 proposal 提案(创建 /dsjjh)到每个 follower ,每个 follower 记录下要创建 /dsjjh
4 超过半数 quorum(参与者,包括 leader 自己)同意提案,则leader commit提案,leader本地先创建 /dsjjh节点
5 leader 通知所有 follower 也 commit 提案,follower 在本地创建 /dsjjh
6 follower 01 响应 client
读过程:
读过程比较简单,过程如下:
1 客户端先与某个 ZK 服务器建立 Session
2 然后 直接从此 ZK 服务器读取数据,并返回客户端
3 关闭session
02 安内 :图说 ZK 集群 leader选举
zk 服务器共有 4 种状态-- Looking followering leadering observing。
集群 leader 选举 分为 全新集群选举 和 非全新集群选举,这两种方式都遵循内部的选举公式。
Q:
什么是选举公式?
A:
选举公式是 ZK 集群内部选举 leader时 遵循的规则。信息选举的格式为 :vote(sid,zxid),其中sid 为 机器编号,zxid 为事务编号。
选举过程中,先判断 zxid ,再判断 sid ,小编号遵从大编号。
全新集群选举
1 server1 启动时 将投票投给自己, 投票信息为 vote(1,0) 没有超过半数
2 server2 启动,server1 投票给自己和其他服务器,server2 投票给自己和其他服务器,server1 收到的自己的投票为 vote(1,0) 和 server2的投票为vote(2,0),server2 收到server1的投票为 vote(1,0) 和 自己的投票 为vote(2,0)
3 处理投票,只能投一票
server1 (1,0)(2,0) --> (2,0)
server2 (1,0) (2,0) --> (2,0)
根据选举公式, zxid相同的,sid大的胜出,server1 和 server2 都保留了 (2,0) 票
4 再次投票选举,server1 和 server2 投票(2,0),投给自己和其他服务器:
server1 有自己的 (2,0)和server2 投票的 (2,0), 两票选举 server2 为leader,server2 有自己的(2,0) 和 server1 投票的 (2,0),两票选举 server2 为leader。
统计投票结果,server1 选举 server2 ,server2选举server2,超过半数 server2 被选举为leader
改变服务器状态,server2由looking改为leadering ,server1由looking改为followering
5 当server3 启动时发现已经有leader ,不再选举 直接从 Looking 改为 following。
非全新集群选举
非全新集群选举与全新类似,如图当 server2 挂掉,重新选举
1 server1 投票给自己 和 server3,server3类似
2 server1 票(1,3)(3,2),server3 得票 (1,3)(3,2)
3 根据选举公司处理得票结果
server1 -- (1,3)
server3 -- (1,3)
4 重新选举 server1 投票给自己和server3,server3类似
server1 票 (1,3)(1,3) -- 选举结果为 server1
server3 票 (1,3)(1,3) -- 选举结果为 server1
超过半数选举server1 ,server1 由looking 改为 leadering。
03 图解攘外安内背后的 ZAB 算法
ZAB 算法
在看 ZAB 算法前,需要先了解下 仲裁 (quorum) 与 网络分区的相关知识。
quorum (仲裁)
在 leader 发起 proposal 提议时,多数派同意即生效,即半数原则,这就是 仲裁 。
quorum 数选择: (集群节点数/2 )+1 ,只有集群大于等于 quorum 数集群才能正常工作
为什么用 quorum 呢,主要有以下两点:
1 不需要所有服务器都响应结果,多数派响应,提议即生效,如果需要所有服务器都响应结果,一旦出现网络原因,可能会造成大量延迟
2 加快集群响应速度,假设有5台zk服务器,只要其中半数 3台同意提案,就能给客户端响应结果
2 网络分区 、脑裂
网络分区:网络通信故障,集群被分成二个部分
脑裂:出现多leader情况。原leader 在一个分区,新分区由于网络分区,又选举出来一个 leader,从而出现2个leader的情况。
ZAB 算法 与 raft 算法类似,它们之间的差别很小,不同点在于:
ZAB 心跳 从 follower 到 leader,raft 相反
ZAB的任期叫 epoch ,raft 叫 term。
下面以 raft 算法为例,来看看算法的实现过程:
点击 “动态演示”,可查看 raft 算法动态演示过程。
raft 算法
图片中 ⚪形状 解释
黑边包围的⚪代表 leader ,有凹边的⚪表示follower ,每个follower 都会倒计时,在倒计时结束前收到 leader 心跳则重新倒计时,若倒计时结束前未收到leader 心跳,则所有follower 开始新一轮leader 选举。
图1 是正常运行的集群,NodeB 为leader (leader最外层是黑圈),集群通过心跳连接
图2 出现网络分区 上面3个节点无法与 下面2个节点通信,由于上面的节点无法通信,在一段时间内没有收到心跳,状态由followeing 变为 looking,开始选举leader
图3 上面三个节点 NodeE 选举为 leader ,同时由于网络分区, 原来的 Node B也是leader, 出现多主的情况
图4 此时一个客户端连接上 NodeB 发送写请求,NodeB 将写请求提案, NodeA 同意提案,由于 此集群 quorum 数为 3,但只有 NodeB、NodeA两台机器同意,所以此次写请求不会被提交;类似的 当另一个客户端连接 NodeE 发送写请求,由于提案通过数为 3,所以写的请求通过
图5 网络分区消失后,各个节点能相互通信,此时第一任leader 发送心跳,第二人leader发送心跳,当 原leader NodeB 收到信Leader NodeE的心跳信息,发现NodeE 任期 term 为 2 ,出现了新的leader ,自己降级为 follower
图6 下面的 follower NodeA,NodeB 收到新leader 心跳 将任务 term 改成2 并进行状态同步,同步第二任 leader 的状态。
04 ZK 集群状态同步
Q:
为什么需要状态同步?
A:
完成 leader 选举后,zk 集群进入状态同步阶段,之所以进行同步是因为新的 leader 刚刚选举出来后,follower 上的节点信息有可能与 leader 不一致,经过状态同步后使得集群 follower 节点 与 leader 节点数据保持一致。
ZK 状态同步
1 leader 发送 newLeader 包到 follower 节点
2 follower 节点收到包后比较 zxid 是否比 leader 的zxid小,如果小则进行状态同步
3 leader 开启 followerlearner 线程,给需要同步的 每个 follower线程,进行数据同步
4 leader 主线程等待 follower线程处理结果
5 半数 follower 线程状态同步成功后,leader 才开始对外服务。
followerlearner 线程逻辑:
1 接收 follower的最大 zxid
2 将 follower 最大zxid 与 leader max zxid 比较
如果相等 则说明follower zxid 是最新的
如果follower zxid 小(或者大),则需要与 leader zxid 保持同步
3 以上消息完毕之后,发送UPTODATE封包告知follower当前数据就是最新的了。
05 ZK 分布式锁的应用
Q:
为什么用到分布式锁?
A:
为了保证一个方法在高并发情况下的同一时间只能被同一个线程执行,在分布式系统中,由于分布式系统多线程、多进程并且分布在不同机器上,这将使原单机部署情况下的并发控制锁策略失效,为了解决这个问题就需要一种跨JVM的互斥机制来控制共享资源的访问,这就是分布式锁要解决的问题。
ZK 实现分布式锁原理浅析
1 分布式锁使用的 是 临时顺序节点
2 在 ZK 某个持久节点下,进行判断如图的 /lockers
3 此时有3个客户端,争抢锁,同时在 /locks 下创建临时有序节点
client1 --- 创建节点 node_1
client2 --- 创建节点 node_2
client3 --- 创建节点 node_3
4 获取锁时 判断 所有的临时节点 哪个最小,则哪个客户端获取到锁,执行对应逻辑,逻辑执行后需要释放锁 即将对应临时节点删除
5 如果获取锁失败即不是最小节点,则往本身节点的前一个节点注册监听器,即如果是 node_2 节点则监听 node_1,如果是 node_3 节点则监听 node_2,
在回调函数中 设置监听逻辑 ,如果最小节点释放锁,最小节点被删除的同时, 触发监听器,再去判断是否是最小节点,执行获取锁的逻辑,后释放锁,
6最终每个客户端都会获取到锁,创建的临时有序节点最终也都会被删除。
ZK 实现分布式锁的代码关注公众号 “大数据江湖” 回复:“分布式锁”,即可领取。
本文我们通过图文的方式了解了 ZK 集群的相关工作原理,虽然 ZK 相关的维护在日常工作中我们涉及的比较少,但是也正是由于它是协调全局的管理者,理解其相关原理也就变得格外重要。另外文末的“分布式锁”代码不要忘记关注领取哦 ~
— THE END—
ZooKeeper - 攘外安内的多面管理员的更多相关文章
- zookeeper运维 --【】转】
from:http://blog.csdn.net/hengyunabc/article/details/19006911 zookeeper运维 尽管zookeeper在编程上有很多的阱陷,AP ...
- zookeeper 运维
尽管zookeeper在编程上有很多的阱陷,API也非常的难用,但zookeeper服务本身可以说是很牢靠的了,所以在网上貌似关于运维的文章比较少. 但省心并不代表不会出麻烦,下面总结下zookeep ...
- [转]Zookeeper系列(一)
一.ZooKeeper的背景 1.1 认识ZooKeeper ZooKeeper---译名为“动物园管理员”.动物园里当然有好多的动物,游客可以根据动物园提供的向导图到不同的场馆观赏各种类型的动物,而 ...
- ZooKeeper文档(二)
ZooKeeper:因为协调的分布式系统是一个动物园 ZooKeeper对分布式应用来说是一个高性能的协调服务.它暴露通常的服务-比如命名,配置管理,同步,和组服务-用一种简单的接口,所以你不用从头开 ...
- Zookeeper系列(一)
一.ZooKeeper的背景 1.1 认识ZooKeeper ZooKeeper---译名为“动物园管理员”.动物园里当然有好多的动物,游客可以根据动物园提供的向导图到不同的场馆观赏各种类型的动物,而 ...
- ZooKeeper的ACL实现源码阅读
什么是ACL(Access Control List) zookeeper在分布式系统中承担中间件的作用,它管理的每一个节点上可能都存储这重要的信息,因为应用可以读取到任意节点,这就可能造成安全问题, ...
- 服务发现对比:Zookeeper vs etcd vs Consul
我们拥有的服务越多,如果我们使用预定义的端口,就会发生冲突的可能性越大.毕竟,在同一端口上不能监听两个服务.管理一百个服务所使用的所有端口的紧密列表本身就是一项挑战.将那些服务所需的数据库添加到该列表 ...
- 大数据之Zookeeper概述
Zookeeper概述 Zookeeper是一个开放源码的分布式应用程序协调服务,是 Google的Chubby一个开源的实现,是 Hadoop和 HBASE的重要组件.主要解决分布式应用一致性问题. ...
- Zookeeper分布式过程协同技术 - 概念及基础
Zookeeper分布式过程协同技术 - 概念及基础 Zookeeper是什么? Zookeeper是一种分布式过程协同技术,其所提供的客户端API功能强大,其中包括: 保障强一致性.有序性和持久性. ...
随机推荐
- C++Primer 5th Chap8 The IO Library
IO类: 头文件 类型 iostream istream,从流读取数据 ostream,向流写入数据 iostream,读写流 fstream ifstream,从文件读取数据,默认in模式打开 of ...
- 1,electron搭建
electron可以用js来创建丰富的桌面应用 一个electron应用的目录结构 your-app/ ├── package.json ├── main.js └── index.html 1,安装 ...
- 【数学】Prime-Factor Prime
Prime-Factor Prime 题目描述 A positive integer is called a "prime-factor prime" when the numbe ...
- 关于MySQL的驱动org.gjt.mm.mysql.Driver
今天看了一个比较老视频使用org.gjt.mm.mysql.Driver来驱动连接,便试了一下看看怎么样,结果一直连不上数据库,后来看了tomcat的后台发现有报这个问题,于是把驱动改成com.mys ...
- MySQL AND 和 OR 联合使用带来的坑
MySQL 基础篇 三范式 MySQL 军规 MySQL 配置 MySQL 用户管理和权限设置 MySQL 常用函数介绍 MySQL 字段类型介绍 MySQL 多列排序 MySQL 行转列 列转行 M ...
- 怎样修改原型对象prototype
修改原型对象的方法分为两种情况, 一种是对原型对象的属性方法做增删改, 一种改变原型对象的指向. 第一种: 对原型对象的属性/方法做增删改 function Person(name){ this.na ...
- ASP.NET MVC或者.net Core mvc 页面使用富文本控件的 保存问题
https://blog.csdn.net/leftfist/article/details/69629394 目前在做的项目存在XSS安全漏洞! 原因是有一些页面使用了富文本编辑框,为了使得其内容可 ...
- 转录组测序(RNA-seq)技术
转录组是某个物种或者特定细胞类型产生的所有转录本的集合.转录组研究能够从整体水 平研究基因功能以及基因结构,揭示特定生物学过程以及疾病发生过程中的分子机理,已广泛应 用于基础研究.临床诊断和药 ...
- “df: cannot read table of mounted file systems”.
“df: cannot read table of mounted file systems”.“df -l” returned an error: “df: cannot read table of ...
- AlterDialog对话框的使用
第一步先写出layout文件 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xml ...