摘要: 该文已加入开源文档:JavaGuide(一份涵盖大部分Java程序员所须要掌握的核心知识)。地址:github.com/Snailclimb/…. 前言 相信你们对 ZooKeeper 应该不算陌生。node
前言git
相信你们对 ZooKeeper 应该不算陌生。可是你真的了解 ZooKeeper 是个什么东西吗?若是别人/面试官让你给他讲讲 ZooKeeper 是个什么东西,你能回答到什么地步呢?程序员
我本人曾经使用过 ZooKeeper 做为 Dubbo 的注册中心,另外在搭建 solr 集群的时候,我使用到了 ZooKeeper 做为 solr 集群的管理工具。前几天,总结项目经验的时候,我忽然问本身 ZooKeeper 究竟是个什么东西?想了半天,脑海中只是简单的能浮现出几句话:“①Zookeeper 能够被用做注册中心。 ②Zookeeper 是 Hadoop 生态系统的一员;③构建 Zookeeper 集群的时候,使用的服务器最好是奇数台。” 可见,我对于 Zookeeper 的理解仅仅是停留在了表面。github
因此,经过本文,但愿带你们稍微详细的了解一下 ZooKeeper 。若是没有学过 ZooKeeper ,那么本文将会是你进入 ZooKeeper 大门的垫脚砖。若是你已经接触过 ZooKeeper ,那么本文将带你回顾一下 ZooKeeper 的一些基础概念。面试
最后,本文只涉及 ZooKeeper 的一些概念,并不涉及 ZooKeeper 的使用以及 ZooKeeper 集群的搭建。 网上有介绍 ZooKeeper 的使用以及搭建 ZooKeeper 集群的文章,你们有须要能够自行查阅。算法
一 什么是 ZooKeeper服务器
ZooKeeper 的由来网络
下面这段内容摘自《从Paxos到Zookeeper 》第四章第一节的某段内容,推荐你们阅读如下:session
Zookeeper最先起源于雅虎研究院的一个研究小组。在当时,研究人员发现,在雅虎内部不少大型系统基本都须要依赖一个相似的系统来进行分布式协调,可是这些系统每每都存在分布式单点问题。因此,雅虎的开发人员就试图开发一个通用的无单点问题的分布式协调框架,以便让开发人员将精力集中在处理业务逻辑上。数据结构
关于“ZooKeeper”这个项目的名字,其实也有一段趣闻。在立项初期,考虑到以前内部不少项目都是使用动物的名字来命名的(例如著名的Pig项目),雅虎的工程师但愿给这个项目也取一个动物的名字。时任研究院的首席科学家RaghuRamakrishnan开玩笑地说:“在这样下去,咱们这儿就变成动物园了!”此话一出,你们纷纷表示就叫动物园管理员吧一一一由于各个以动物命名的分布式组件放在一块儿,雅虎的整个分布式系统看上去就像一个大型的动物园了,而Zookeeper正好要用来进行分布式环境的协调一一因而,Zookeeper的名字也就由此诞生了。
1.1 ZooKeeper 概览
ZooKeeper 是一个开源的分布式协调服务,ZooKeeper框架最初是在“Yahoo!"上构建的,用于以简单而稳健的方式访问他们的应用程序。 后来,Apache ZooKeeper成为Hadoop,HBase和其余分布式框架使用的有组织服务的标准。 例如,Apache HBase使用ZooKeeper跟踪分布式数据的状态。ZooKeeper 的设计目标是将那些复杂且容易出错的分布式一致性服务封装起来,构成一个高效可靠的原语集,并以一系列简单易用的接口提供给用户使用。
原语: 操做系统或计算机网络用语范畴。是由若干条指令组成的,用于完成必定功能的一个过程。具备不可分割性·即原语的执行必须是连续的,在执行过程当中不容许被中断。
ZooKeeper 是一个典型的分布式数据一致性解决方案,分布式应用程序能够基于 ZooKeeper 实现诸如数据发布/订阅、负载均衡、命名服务、分布式协调/通知、集群管理、Master 选举、分布式锁和分布式队列等功能。
Zookeeper 一个最经常使用的使用场景就是用于担任服务生产者和服务消费者的注册中心。 服务生产者将本身提供的服务注册到Zookeeper中心,服务的消费者在进行服务调用的时候先到Zookeeper中查找服务,获取到服务生产者的详细信息以后,再去调用服务生产者的内容与数据。以下图所示,在 Dubbo架构中 Zookeeper 就担任了注册中心这一角色。
1.2 结合我的使用状况的讲一下 ZooKeeper
在我本身作过的项目中,主要使用到了 ZooKeeper 做为 Dubbo 的注册中心(Dubbo 官方推荐使用 ZooKeeper注册中心)。另外在搭建 solr 集群的时候,我使用 ZooKeeper 做为 solr 集群的管理工具。这时,ZooKeeper 主要提供下面几个功能:一、集群管理:容错、负载均衡。二、配置文件的集中管理三、集群的入口。
我我的以为在使用 ZooKeeper 的时候,最好是使用 集群版的 ZooKeeper 而不是单机版的。官网给出的架构图就描述的是一个集群版的 ZooKeeper 。一般 3 台服务器就能够构成一个 ZooKeeper 集群了。
为何最好使用奇数台服务器构成 ZooKeeper 集群?
咱们知道在Zookeeper中 Leader 选举算法采用了Zab协议。Zab核心思想是当多数 Server 写成功,则任务数据写成功。
①若是有3个Server,则最多容许1个Server 挂掉。
②若是有4个Server,则一样最多容许1个Server挂掉。
既然3个或者4个Server,一样最多容许1个Server挂掉,那么它们的可靠性是同样的,因此选择奇数个ZooKeeper Server便可,这里选择3个Server。12341234
二 关于 ZooKeeper 的一些重要概念
2.1 重要概念总结
● ZooKeeper 自己就是一个分布式程序(只要半数以上节点存活,ZooKeeper 就能正常服务)。
● 为了保证高可用,最好是以集群形态来部署 ZooKeeper,这样只要集群中大部分机器是可用的(可以容忍必定的机器故障),那么 ZooKeeper 自己仍然是可用的。
● ZooKeeper 将数据保存在内存中,这也就保证了 高吞吐量和低延迟(可是内存限制了可以存储的容量不太大,此限制也是保持znode中存储的数据量较小的进一步缘由)。
● ZooKeeper 是高性能的。 在“读”多于“写”的应用程序中尤为地高性能,由于“写”会致使全部的服务器间同步状态。(“读”多于“写”是协调服务的典型场景。)
● ZooKeeper有临时节点的概念。 当建立临时节点的客户端会话一直保持活动,瞬时节点就一直存在。而当会话终结时,瞬时节点被删除。持久节点是指一旦这个ZNode被建立了,除非主动进行ZNode的移除操做,不然这个ZNode将一直保存在Zookeeper上。
● ZooKeeper 底层其实只提供了两个功能:①管理(存储、读取)用户程序提交的数据;②为用户程序提交数据节点监听服务。
下面关于会话(Session)、 Znode、版本、Watcher、ACL概念的总结都在《从Paxos到Zookeeper 》第四章第一节以及第七章第八节有提到,感兴趣的能够看看!
2.2 会话(Session)
Session 指的是 ZooKeeper 服务器与客户端会话。在 ZooKeeper 中,一个客户端链接是指客户端和服务器之间的一个 TCP 长链接。客户端启动的时候,首先会与服务器创建一个 TCP 链接,从第一次链接创建开始,客户端会话的生命周期也开始了。经过这个链接,客户端可以经过心跳检测与服务器保持有效的会话,也可以向Zookeeper服务器发送请求并接受响应,同时还可以经过该链接接收来自服务器的Watch事件通知。 Session的sessionTimeout值用来设置一个客户端会话的超时时间。当因为服务器压力太大、网络故障或是客户端主动断开链接等各类缘由致使客户端链接断开时,只要在sessionTimeout规定的时间内可以从新链接上集群中任意一台服务器,那么以前建立的会话仍然有效。
在为客户端建立会话以前,服务端首先会为每一个客户端都分配一个sessionID。因为 sessionID 是 Zookeeper 会话的一个重要标识,许多与会话相关的运行机制都是基于这个 sessionID 的,所以,不管是哪台服务器为客户端分配的 sessionID,都务必保证全局惟一。
2.3 Znode
在谈到分布式的时候,咱们一般说的“节点"是指组成集群的每一台机器。然而,在Zookeeper中,“节点"分为两类,第一类一样是指构成集群的机器,咱们称之为机器节点;第二类则是指数据模型中的数据单元,咱们称之为数据节点一一ZNode。
Zookeeper将全部数据存储在内存中,数据模型是一棵树(Znode Tree),由斜杠(/)的进行分割的路径,就是一个Znode,例如/foo/path1。每一个上都会保存本身的数据内容,同时还会保存一系列属性信息。
在Zookeeper中,node能够分为持久节点和临时节点两类。所谓持久节点是指一旦这个ZNode被建立了,除非主动进行ZNode的移除操做,不然这个ZNode将一直保存在Zookeeper上。而临时节点就不同了,它的生命周期和客户端会话绑定,一旦客户端会话失效,那么这个客户端建立的全部临时节点都会被移除。另外,ZooKeeper还容许用户为每一个节点添加一个特殊的属性:SEQUENTIAL.一旦节点被标记上这个属性,那么在这个节点被建立的时候,Zookeeper会自动在其节点名后面追加上一个整型数字,这个整型数字是一个由父节点维护的自增数字。
2.4 版本
在前面咱们已经提到,Zookeeper 的每一个 ZNode 上都会存储数据,对应于每一个ZNode,Zookeeper 都会为其维护一个叫做 Stat 的数据结构,Stat中记录了这个 ZNode 的三个数据版本,分别是version(当前ZNode的版本)、cversion(当前ZNode子节点的版本)和 cversion(当前ZNode的ACL版本)。
2.5 Watcher
Watcher(事件监听器),是Zookeeper中的一个很重要的特性。Zookeeper容许用户在指定节点上注册一些Watcher,而且在一些特定事件触发的时候,ZooKeeper服务端会将事件通知到感兴趣的客户端上去,该机制是Zookeeper实现分布式协调服务的重要特性。
2.6 ACL
Zookeeper采用ACL(AccessControlLists)策略来进行权限控制,相似于 UNIX 文件系统的权限控制。Zookeeper 定义了以下5种权限。
其中尤为须要注意的是,CREATE和DELETE这两种权限都是针对子节点的权限控制。
三 ZooKeeper 特色
● 顺序一致性: 从同一客户端发起的事务请求,最终将会严格地按照顺序被应用到 ZooKeeper 中去。
● 原子性: 全部事务请求的处理结果在整个集群中全部机器上的应用状况是一致的,也就是说,要么整个集群中全部的机器都成功应用了某一个事务,要么都没有应用。
● 单一系统映像 : 不管客户端连到哪个 ZooKeeper 服务器上,其看到的服务端数据模型都是一致的。
● 可靠性: 一旦一次更改请求被应用,更改的结果就会被持久化,直到被下一次更改覆盖。
四 ZooKeeper 设计目标
4.1 简单的数据模型
ZooKeeper 容许分布式进程经过共享的层次结构命名空间进行相互协调,这与标准文件系统相似。 名称空间由 ZooKeeper 中的数据寄存器组成 - 称为znode,这些相似于文件和目录。 与为存储设计的典型文件系统不一样,ZooKeeper数据保存在内存中,这意味着ZooKeeper能够实现高吞吐量和低延迟。
4.2 可构建集群
为了保证高可用,最好是以集群形态来部署 ZooKeeper,这样只要集群中大部分机器是可用的(可以容忍必定的机器故障),那么zookeeper自己仍然是可用的。 客户端在使用 ZooKeeper 时,须要知道集群机器列表,经过与集群中的某一台机器创建 TCP 链接来使用服务,客户端使用这个TCP连接来发送请求、获取结果、获取监听事件以及发送心跳包。若是这个链接异常断开了,客户端能够链接到另外的机器上。
ZooKeeper 官方提供的架构图:
上图中每个Server表明一个安装Zookeeper服务的服务器。组成 ZooKeeper 服务的服务器都会在内存中维护当前的服务器状态,而且每台服务器之间都互相保持着通讯。集群间经过 Zab 协议(Zookeeper Atomic Broadcast)来保持数据的一致性。
4.3 顺序访问
对于来自客户端的每一个更新请求,ZooKeeper 都会分配一个全局惟一的递增编号,这个编号反应了全部事务操做的前后顺序,应用程序可使用 ZooKeeper 这个特性来实现更高层次的同步原语。 这个编号也叫作时间戳——zxid(Zookeeper Transaction Id)
4.4 高性能
ZooKeeper 是高性能的。 在“读”多于“写”的应用程序中尤为地高性能,由于“写”会致使全部的服务器间同步状态。(“读”多于“写”是协调服务的典型场景。)
五 ZooKeeper 集群角色介绍
最典型集群模式: Master/Slave 模式(主备模式)。在这种模式中,一般 Master服务器做为主服务器提供写服务,其余的 Slave 服务器从服务器经过异步复制的方式获取 Master 服务器最新的数据提供读服务。
可是,在 ZooKeeper 中没有选择传统的 Master/Slave 概念,而是引入了Leader、Follower 和 Observer 三种角色。以下图所示
ZooKeeper 集群中的全部机器经过一个 Leader 选举过程来选定一台称为 “Leader” 的机器,Leader 既能够为客户端提供写服务又能提供读服务。除了 Leader 外,Follower 和 Observer 都只能提供读服务。Follower 和 Observer 惟一的区别在于 Observer 机器不参与 Leader 的选举过程,也不参与写操做的“过半写成功”策略,所以 Observer 机器能够在不影响写性能的状况下提高集群的读性能。
六 ZooKeeper &ZAB 协议&Paxos算法
6.1 ZAB 协议&Paxos算法
Paxos 算法应该能够说是 ZooKeeper 的灵魂了。可是,ZooKeeper 并无彻底采用 Paxos算法 ,而是使用 ZAB 协议做为其保证数据一致性的核心算法。另外,在ZooKeeper的官方文档中也指出,ZAB协议并不像 Paxos 算法那样,是一种通用的分布式一致性算法,它是一种特别为Zookeeper设计的崩溃可恢复的原子消息广播算法。
6.2 ZAB 协议介绍
ZAB(ZooKeeper Atomic Broadcast 原子广播) 协议是为分布式协调服务 ZooKeeper 专门设计的一种支持崩溃恢复的原子广播协议。 在 ZooKeeper 中,主要依赖 ZAB 协议来实现分布式数据一致性,基于该协议,ZooKeeper 实现了一种主备模式的系统架构来保持集群中各个副本之间的数据一致性。
6.3 ZAB 协议两种基本的模式:崩溃恢复和消息广播
ZAB协议包括两种基本的模式,分别是 崩溃恢复和消息广播。当整个服务框架在启动过程当中,或是当 Leader 服务器出现网络中断、崩溃退出与重启等异常状况时,ZAB 协议就会进人恢复模式并选举产生新的Leader服务器。当选举产生了新的 Leader 服务器,同时集群中已经有过半的机器与该Leader服务器完成了状态同步以后,ZAB协议就会退出恢复模式。其中,所谓的状态同步是指数据同步,用来保证集群中存在过半的机器可以和Leader服务器的数据状态保持一致。
当集群中已经有过半的Follower服务器完成了和Leader服务器的状态同步,那么整个服务框架就能够进人消息广播模式了。 当一台一样遵照ZAB协议的服务器启动后加人到集群中时,若是此时集群中已经存在一个Leader服务器在负责进行消息广播,那么新加人的服务器就会自觉地进人数据恢复模式:找到Leader所在的服务器,并与其进行数据同步,而后一块儿参与到消息广播流程中去。正如上文介绍中所说的,ZooKeeper设计成只容许惟一的一个Leader服务器来进行事务请求的处理。Leader服务器在接收到客户端的事务请求后,会生成对应的事务提案并发起一轮广播协议;而若是集群中的其余机器接收到客户端的事务请求,那么这些非Leader服务器会首先将这个事务请求转发给Leader服务器。
关于 ZAB 协议&Paxos算法 须要讲和理解的东西太多了,说实话,笔主到如今不太清楚这俩兄弟的具体原理和实现过程。推荐阅读下面两篇文章:
● 图解 Paxos 一致性协议
● Zookeeper ZAB 协议分析
关于如何使用 zookeeper 实现分布式锁,能够查看下面这篇文章:
● 10分钟看懂!基于Zookeeper的分布式锁
六 总结
经过阅读本文,想必你们已从 ①ZooKeeper的由来。 -> ②ZooKeeper 究竟是什么 。-> ③ ZooKeeper 的一些重要概念(会话(Session)、 Znode、版本、Watcher、ACL)-> ④ZooKeeper 的特色。 -> ⑤ZooKeeper 的设计目标。-> ⑥ ZooKeeper 集群角色介绍 (Leader、Follower 和 Observer 三种角色)-> ⑦ZooKeeper &ZAB 协议&Paxos算法。 这七点了解了 ZooKeeper 。
写在最后:
码字不易看到最后了,那就点个关注呗,只收藏不点关注的都是在耍流氓!
关注并私信我“架构”,免费送一些Java架构资料,先到先得!