zookeeper源码 — 五、处理写请求过程

摘要:
目录处理写请求的整个过程客户端发起写请求跟随者和领导者交互跟随者向客户端发送请求处理写请求zk的整个过程确保了分布式数据的一致性。

目录

  • 处理写请求总体过程
  • 客户端发起写请求
  • follower和leader交互过程
  • follower发送请求给客户端

处理写请求总体过程

zk为了保证分布式数据一致性,使用ZAB协议,在客户端发起一次写请求的时候时候,假设该请求请求到的是follower,follower不会直接处理这个请求,而是转发给leader,由leader发起投票决定该请求最终能否执行成功,所以整个过程client、被请求的follower、leader、其他follower都参与其中。以创建一个节点为例,总体流程如下

zookeeper源码 — 五、处理写请求过程第1张

从图中可以看出,创建流程为

  1. follower接受请求,解析请求
  2. follower通过FollowerRequestProcessor将请求转发给leader
  3. leader接收请求
  4. leader发送proposal给follower
  5. follower收到请求记录txlog、snapshot
  6. follower发送ack给leader
  7. leader收到ack后进行commit,并且通知所有的learner,发送commit packet给所有的learner

这里说的follower、leader都是server,在zk里面server总共有这么几种

zookeeper源码 — 五、处理写请求过程第2张

由于server角色不同,对于请求所做的处理不同,每种server包含的processor也不同,下面细说下具体有哪些processor。

follower的processor链

这里的follower是FollowerZooKeeperServer,通过setupRequestProcessors来设置自己的processor链

FollowerRequestProcessor -> CommitProcessor ->FinalRequestProcessor

每个processor对应的功能为:

FollowerRequestProcessor:

作用:将请求先转发给下一个processor,然后根据不同的OpCode做不同的操作

如果是:sync,先加入org.apache.zookeeper.server.quorum.FollowerZooKeeperServer#pendingSyncs,然后发送给leader

如果是:create等,直接转发

如果是:createSession或者closeSession,如果不是localsession则转发给leader

CommitProcessor :

有一个WorkerService,将请求封装为CommitWorkRequest执行

作用:

转发请求,读请求直接转发给下一个processor

写请求先放在pendingRequests对应的sessionId下的list中,收到leader返回的commitRequest再处理

  1. 处理读请求(不会改变服务器状态的请求)
  2. 处理committed的写请求(经过leader 处理完成的请求)

维护一个线程池服务WorkerService,每个请求都在单独的线程中处理

  1. 每个session的请求必须按顺序执行
  2. 写请求必须按照zxid顺序执行
  3. 确认一个session中写请求之间没有竞争

FinalRequestProcessor:

总是processor chain上最后一个processor

作用:

  1. 实际处理事务请求的processor
  2. 处理query请求
  3. 返回response给客户端

SyncRequestProcessor:

作用:

  1. 接收leader的proposal进行处理
  2. 从org.apache.zookeeper.server.SyncRequestProcessor#queuedRequests中取出请求记录txlog和snapshot
  3. 然后加入toFlush,从toFlush中取出请求交给org.apache.zookeeper.server.quorum.SendAckRequestProcessor#flush处理

leader的processor链

这里的leader就是LeaderZooKeeperServer

通过setupRequestProcessors来设置自己的processor链

PrepRequestProcessor -> ProposalRequestProcessor ->CommitProcessor -> Leader.ToBeAppliedRequestProcessor ->FinalRequestProcessor

客户端发起写请求

在客户端启动的时候会创建Zookeeper实例,client会连接到server,后面client在创建节点的时候就可以直接和server通信,client发起创建创建节点请求的过程是:

org.apache.zookeeper.ZooKeeper#create(java.lang.String, byte[], java.util.List<org.apache.zookeeper.data.ACL>, org.apache.zookeeper.CreateMode)
org.apache.zookeeper.ClientCnxn#submitRequest
org.apache.zookeeper.ClientCnxn#queuePacket
  1. ZooKeeper#create方法中构造请求的request
  2. ClientCnxn#queuePacket方法中将request封装到packet中,将packet放入发送队列outgoingQueue中等待发送
    1. SendThread不断从发送队列outgoingQueue中取出packet发送
  3. 通过 packet.wait等待server返回

follower和leader交互过程

client发出请求后,follower会接收并处理该请求。选举结束后follower确定了自己的角色为follower,一个端口和client通信,一个端口和leader通信。监听到来自client的连接口建立新的session,监听对应的socket上的读写事件,如果client有请求发到follower,follower会用下面的方法处理

org.apache.zookeeper.server.NIOServerCnxn#readPayload
org.apache.zookeeper.server.NIOServerCnxn#readRequest

readPayload这个方法里面会判断是连接请求还是非连接请求,连接请求在之前session建立的文章中介绍过,这里从处理非连接请求开始。

follower接收client请求

follower收到请求之后,先请求请求的opCode类型(这里是create)构造对应的request,然后交给第一个processor执行,follower的第一个processor是FollowerRequestProcessor.

follower转发请求给leader

由于在zk中follower是不能处理写请求的,需要转交给leader处理,在FollowerRequestProcessor中将请求转发给leader,转发请求的调用堆栈是

serialize(OutputArchive, String):82, QuorumPacket (org.apache.zookeeper.server.quorum), QuorumPacket.java
 writeRecord(Record, String):123, BinaryOutputArchive (org.apache.jute), BinaryOutputArchive.java
 writePacket(QuorumPacket, boolean):139, Learner (org.apache.zookeeper.server.quorum), Learner.java
 request(Request):191, Learner (org.apache.zookeeper.server.quorum), Learner.java
 run():96, FollowerRequestProcessor (org.apache.zookeeper.server.quorum), FollowerRequestProcessor.java

FollowerRequestProcessor是一个线程在zk启动的时候就开始运行,主要逻辑在run方法里面,run方法的主要逻辑是

zookeeper源码 — 五、处理写请求过程第3张

先把请求提交给CommitProcessor(后面leader发送给follower的commit请求对应到这里),然后将请求转发给leader,转发给leader的过程就是构造一个QuorumPacket,通过之前选举通信的端口发送给leader。

leader接收follower请求

leader获取leader地位以后,启动learnhandler,然后一直在LearnerHandler#run循环,接收来自learner的packet,处理流程是:

processRequest(Request):1003, org.apache.zookeeper.server.PrepRequestProcessor.java
submitLearnerRequest(Request):150, org.apache.zookeeper.server.quorum.LeaderZooKeeperServer.java
run():625, org.apache.zookeeper.server.quorum.LearnerHandler.java

handler判断是REQUEST请求的话交给leader的processor链处理,将请求放入org.apache.zookeeper.server.PrepRequestProcessor#submittedRequests,即leader的第一个processor。这个processor也是一个线程,从submittedRequests中不断拿出请求处理

zookeeper源码 — 五、处理写请求过程第4张

processor主要做了:

  1. 交给CommitProcessor等待提交
  2. 交给leader的下一个processor处理:ProposalRequestProcessor

leader 发送proposal给follower

ProposalRequestProcessor主要作用就是讲请求交给下一个processor并且发起投票,将proposal发送给所有的follower。

// org.apache.zookeeper.server.quorum.Leader#sendPacket
void sendPacket(QuorumPacket qp) {
    synchronized (forwardingFollowers) {
        // 所有的follower,observer没有投票权
        for (LearnerHandler f : forwardingFollowers) {
            f.queuePacket(qp);
        }
    }
}

follower 收到proposal

follower处理proposal请求的调用堆栈

processRequest(Request):214, org.apache.zookeeper.server.SyncRequestProcessor.java
 logRequest(TxnHeader, Record):89, org.apache.zookeeper.server.quorumFollowerZooKeeperServer.java
 processPacket(QuorumPacket):147, org.apache.zookeeper.server.quorum.Follower.java
 followLeader():102, org.apache.zookeeper.server.quorum.Follower.java
 run():1199, org.apache.zookeeper.server.quorum.QuorumPeer.java

将请求放入org.apache.zookeeper.server.SyncRequestProcessor#queuedRequests

follower 发送ack

线程SyncRequestProcessor#run从org.apache.zookeeper.server.SyncRequestProcessor#toFlush中取出请求flush,处理过程

zookeeper源码 — 五、处理写请求过程第5张

  1. follower开始commit,记录txLog和snapShot
  2. 发送commit成功请求给leader,也就是follower给leader的ACK

leader收到ack

leader 收到ack后判断是否收到大多数的follower的ack,如果是说明可以commit,commit后同步给follower
zookeeper源码 — 五、处理写请求过程第6张

follower收到commit

还是Follower#followLeader里面的while循环收到leader的commit请求后,调用下面的方法处理

org.apache.zookeeper.server.quorum.FollowerZooKeeperServer#commit

zookeeper源码 — 五、处理写请求过程第7张

最终加入CommitProcessor.committedRequests队列,CommitProcessor主线程发现队列不空表明需要把这个request转发到下一个processor

follower发送请求给客户端

follower的最后一个processor是FinalRequestProcessor,最后会创建对应的节点并且构造response返回给client

总结

本篇文章主要介绍了client发起一次写请求,client、follower和leader各自的处理过程。当然了,为了简单,其中设定了一些具体的场景,比如请求是发送到follower的而不是leader。

免责声明:文章转载自《zookeeper源码 — 五、处理写请求过程》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇使用VS2008开发OPC客户端程序Spring配置中的"classpath:"与"classpath*:"的区别研究(转)下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

Fiddler4抓包工具使用教程一

本文参考自http://blog.csdn.net/ohmygirl/article/details/17846199,纯属读书笔记,加深记忆 1、抓包工具有很多,为什么要使用Fiddler呢?原因如下: a.Firebug虽然可以抓包,但是对于分析http请求的详细信息,不够强大。模拟http请求的功能也不够,且firebug常常是需要“无刷新修改”,如...

Chrome浏览器F12开发者工具简单使用

                1.如何调出开发者工具 按F12调出 右键检查(或快捷键Ctrl+Shift+i)调出  2.开发者工具初步介绍 chrome开发者工具最常用的四个功能模块:元素(ELements)、控制台(Console)、源代码(Sources),网络(Network)。 元素(Elements):用于查看或修改HTML元素的...

Fiddler修改web端ajax响应结果,方便测试展示

1、下载安装Fiddler,版本如下; 2、安装完成后启动Fiddler并设置代理; (1)选择Tools->Options (2)打开Options后,选择HTTPS; (3)切换至Connections;  3、代理设置完成,访问目标网站,找到要拦截的请求;  4、选中要拦截的请求,点击鼠标右键,选择Save->Response-...

Restful API 设计参考原则

在项目中,需要为后台服务撰写API。刚开始接触的时候,并没有考虑太多,就想提供URL,服务端通过该URL进行查询、创建、更新等操作即可。但再对相关规范进行了解后,才发现,API的设计并没有那么简单,远远不是URL的问题,而是一个通信协议的整体架构 1. 使用GET、POST、PUT、DELETE这几种请求模式 请求模式也可以说是动作、数据传输方式,通常我们...

接口抓包测试工具Charles

作用: 1.抓取网络封包 (web/app)    2.断点替换  -- 请求断点               -- 响应断点    3.弱网测试    4.过滤    5.黑名单 Charles的原理: Charles是一款Http代理服务器和Http监视器,当移动端在无线网连接中按要求设置好代理服务器,使所有对网络的请求都经过Charles客户端来转发...

Raft算法原理剖析

一、复制状态机(replicated state machine)   Raft协议可以使得一个集群的服务器组成复制状态机,在详细了解Raft算法之前,我们先来了解一下什么是复制状态机。一个分布式的复制状态机系统由多个复制单元组成,每个复制单元均是一个状态机,它的状态保存在一组状态变量中,状态机的变量只能通过外部命令来改变。简单理解的话,可以想象成是一组服...