Zookeeper核心原理
-
-
怎么保证任务只在一个节点执行
-
如果orderserver1挂了,其他节点如何发现并接替
-
存在共享资源,互斥性、安全性
google 的chubby 是一个分布式锁服务,通过google chubby 来解决分布式协作、master选举等与分布式锁服务相关的问题
-
-
集群方案(leader follower)还能分担请求,既做了高可用,又做高性能
-
-
每个节点的数据是一致的(必须要有leader)
leader master(带中心化的) redis-cluser (无中心化的)
-
集群中的leader 挂了,怎么办?数据怎么恢复?
选举机制?数据恢复
-
如何去保证数据一致性?(分布式事务)
2pc 协议、二阶提交
-
事务询问
-
协调者向所有的参与者发送事务内容,询问是否可以执行事务提交操作,并开始等待各参与者的响应。
-
-
执行事务
-
各个参与者节点执行事务操作,并将undo和redo信息记录到事务日志中,尽量把提交过程中所有消耗时间的操作和准备的提前完成确保后面100%成功提交事务
-
-
各个参与者向协调者反馈事务询问的响应
-
如果各个参与者都成功执行了事务操作,那么就反馈给参与者yes的响应,表示事务可以执行;
-
如果参与者没有成功执行事务,就反馈给协调者no的响应,表示事务不可以执行;
-
2pc 协议的第一个阶段称为“投票阶段”,即各参与者投票表名是否需要继续执行接下去的事务提交操作。
-
阶段二:执行事务提交
在这个阶段,协调者会根据各参与者的反馈情况来决定最终是否可以进行事务提交操作;
两种可能:
-
执行事务
-
中断事务
如果是读请求,就直接从当前节点中读取数据
如果是写请求,那么请求会转发给leader 提交事务,然后leader将事务广播给集群中的follower节点(注意obeserver节点不参与投票),follower 节点给leader 一个ack (ack表示当前的节点是不是能执行这个事务),只要有超过半数节点写入成功,那么写请求就会被提交。集群节点需要(2n+1)
是zookeeper中的整个核心,起到了主导整个集群的作用
-
事务请求的调度和处理
-
保证事务处理的顺序性
follower角色
-
处理客户端的非事务请求,
-
转发事务请求给leader服务器
-
参与事务请求proposal 的投票(需要半数以上服务器通过才能通知leader commit数据; leader发起的提案, 要求follower投票)
-
参与leader节点选举的投票
observer角色
是一个观察者角色
-
了解集群中的状态变化 ,和对这些状态进行同步
-
工作原理和follower节点一样,唯一差别是不参与事务请求的投票,不参与leader选举
-
observer 只提供非事务请求,通常在于不影响集群事务处理能力的前提下,提升集群非事务处理能力
注:
表示奇数节点, zookeeper中要正常对外提供服务的话,它里面有个投票机制,这个机制就是必须要有过半的机器正常工作,并且能够彼此完成通信进行事务投票结果。
支持崩溃恢复的原子广播协议,主要用于数据一致性
zab协议基本模式
-
崩溃恢复(恢复leader节点和恢复数据)
-
原子广播
消息广播过程实际是一个简化版的二阶提交。2pc
-
leader 接收到消息请求后,将消息赋予一个全局唯一的64位自增id(zxid)。zxid大小,实现因果有序的特征。
-
leader 为每一个follower 准备了一个fifo队列,将带有zxid的消息作为一个提案(proposal)分发给所有follower
-
当follower 收到proposal,先把proposal写到磁盘,写入成功后,再向leader 回复一个ack
-
当leader接收到合法数量的ack后,leader 就会向这个follower 发送commit命令,同时会在本地执行该消息。
-
当follower 收到消息的commit以后,会提交该消息。
注:leader 的投票过程,不需要observer 的ack,但是observer必须要同步leader的数据,保证数据的一致性。
-
-
当leader服务器宕机
集群进去崩溃恢复阶段
对于数据恢复来说
-
已经处理的消息不能丢失
-
当leader 收到合法数量的follower 的ack以后,就会向各个follower 广播消息(commit命令),同时自己也会commit 这条事务消息。
-
如果follower节点收到commit命令之前,leader挂了,会导致部分节点收到commit,部分节点没有收到。
-
zab协议需要保证已经处理的消息不能丢失。
-
-
被丢弃的消息不能再次出现
-
当leader收到事务请求,还未发起事务投票之前,leader挂了
zab的设计思想
-
zxid 是最大的
-
如果leader选举算法能够保证新选举出来的leader服务器拥有集群中所有机器最高编号(zxid最大)的事务proposal,那么就可以保证这个新选举出来的leader一定具有已经提交的提案。因为所有提案被commit之前必须有超过半数的follower ack,即必须有超过半数的服务器的事务日志上有该提案的proposal,因此,只要有合法数量的节点正常工作,就必然有一个节点保存了所有被commit消息的proposal状态。
-
-
epoch的概念,每产生一个新的leader,那么新的leader的epoch会+1,zxid 是64位的数据,低32位表示消息计数器(自增),每收到一条消息,这个值+1,新 leader选举后这个值重置为0。这样设计的原因在于,老的leader 挂了以后重启,他不会选举为leader,y因此此时它的zxid肯定小于当前新的leader.当老的 leader 作为 follower 接入新的 leader 后,新的 leader会让它将所有的拥有旧的epoch号的未被commit的proposal清除 .高32位会存储epoch编号
为了保证事务顺序的一致性,zookeeper 采用了递增的事务id号来标识事务。
所有的提议proposal都在被提出的时候加上了zxid.
zxid 是一个64位的数字(低32位和高32位组成)
-
低32位:表示消息计数器
-
高32位 :表示epoch,用来标识 leader 关系是否改变。 每次一个leader被选出来,都会有一个新的epoch(原来的epoch+1),标识当前属于那个leader的统治时期。
注:
leader: 类似,有自己的年号,每次变更都会在前一个年代上加1。
/tmp/zookeeper/version-2 路径会看到一个 currentepoch文件。文件显示的是当前的epoch
通过命令查看事务日志
java -cp :/opt/zookeeper/zookeeper-3.4.10/lib/slf4j-api-1.6.1.jar:/opt/zookeeper/zookeeper-3.4.10/zookeeper-3.4.10.jar org.apache.zookeeper.server.logformatter /tmp/zookeeper/version-2/log.100000001
注:
zxid(事务id)事务id 越大,那么表示数据越新, 最大会设置为leader, ,
epoch ,每一轮投票,epoch 都会递增
myid (服务器id ,server id)
myid 越大,在leader 选举机制中权重越大
服务启动时的状态都是looking,观望状态
leading
following
observing
-
-
投票信息包含(myid,zxid,epoch)
-
-
接受来自各个服务器的投票
-
判断该投票是否有效
-
检查是否来自本轮投票epoch
-
检查是否来时looking状态的服务器
-
-
-
处理投票
-
检查zxid,如果zxid比较大,那么设置为leader,
-
如果zxid相同,会检查myid,myid比较大的,设置为leader
-
-
统计投票
-
判断是否已经有过半机器接受到相同的投票信息
-
如果有过半机器接受,便认为已经选举出了leader
-
-
改变服务器状态
-
如果是follower,那么状态变为following
-
如果是leader,那么状态变为leading
-
变更状态
-
leader 挂后,余下非observer服务器都会将自己的服务器状态变为looking,
-
开始进入leader选举过程
-
-
每个server会发起一个投票。
-
运行期间,zxid 可能不同,然后将各自的投票发送给集群中的所有机器。
-
-
其余与启动时过程相同。
上一篇: 23种设计模式之访问者模式
下一篇: 设计模式系列 - 装饰器模式