ZooKeeper核心原理及應用場景
為什麼會有ZooKeeper
我們知道要寫一個分散式應用是非常困難的,主要原因就是區域性故障。一個訊息透過網路在兩個節點之間傳遞時,網路如果發生故障,傳送方並不知道接收方是否接收到了這個訊息。有可能是收到訊息以後發生了網路故障,也有可能是沒有收到訊息,又或者可能接收方的程式死了。傳送方唯一的確認方法就是再次連線傳送訊息,並向他進行詢問。這就是區域性故障:根本不知道操作是否失敗。因此,大部分分散式應用需要一個主控、協調控制器來管理物理分佈的子程式。所以大部分應用需要開發私有的協調程式,協調程式的反覆編寫浪費時間,這個時候就需要一個通用的、伸縮性好的協調器。就是因為這樣的場景,ZooKeeper應運而生,ZooKeeper的設計目的,就是為了減輕分散式應用程式所承擔的協調任務。
ZooKeeper常用的應用場景
01 / 分散式協調
分散式協調簡單說就是有人對ZooKeeper中的資料做了監聽,如果修改了ZooKeeper中被監聽的資料,ZooKeeper反過來會告訴給發起監聽的人資料的變更。比如在Kafka的設計中,Kafka的一個節點在ZooKeeper中建立了一個資料,Kafka的策略是誰建立了這個資料誰就是Kafka叢集的主節點,其餘的節點都會去監聽這個資料。如果主節點當機了,這ZooKeeper對應的資料就會發生變更,既而監聽這個資料的其餘節點就會感知到主節點當機了,然後重新進行選舉。
02 / 後設資料管理
很多分散式的程式需要集中式的管理自己的後設資料,這個時候ZooKeeper就是一個很好的選擇。比如Kafka,Storm等分散式的工具就會把叢集裡核心的後設資料存放在ZooKeeper中。
03 / 高可用
很多分散式的專案都是主從式的架構,正常情況下叢集裡有一個是主節點,其餘的都是從節點。但是如果只有一個主節點的話,程式就會有單點故障問題,那麼這個時候就需要部署多個主節點實現高可用了,利用ZooKeeper從多個主節點中選出一個作為master,其餘的作為StandBy。比如鼎鼎大名的HDFS就是靠ZooKeeper實現的高可用。
04 / 分散式鎖
在企業裡面很多的專案需要分散式鎖,我們可以使用ZooKeeper搞分散式鎖,不過這兒大家要注意一點,ZooKeeper確實是可以搞分散式鎖的,但是ZooKeeper不支援太高的併發,也就是說如果是高併發的情況下,分散式鎖用ZooKeeper可能也不太適合,如果在高併發的情況下建議大家使用Redis去搞分散式鎖,但是併發不太高的情況下用ZooKeeper搞分散式鎖是比較方便的,也有很多人確實是這麼使用的。
ZooKeeper核心原理
01 / ZooKeeper叢集架構
在ZooKeeper叢集當中,叢集中的伺服器角色分為leader和learner,learner又分為observer和follower,具體功能如下:
0x01、leader(領導者)
為客戶端提供讀和寫的功能,負責投票的發起和決議,叢集裡面只有leader才能接受寫的服務。
0x02、follower(跟隨者)
為客戶端提供讀服務,如果是寫的服務則轉發給leader。在選舉過程中進行投票。
0x03、observer(觀察者)
為客戶端提供讀服務,如果是寫服務就轉發個leader。不參與leader的選舉投票。也不參與寫的過半原則機制。在不影響寫的前提下,提高叢集讀的效能,此角色於zookeeper3.3系列新增的角色。
0x04、client(客戶端)
連線zookeeper叢集的使用者,請求的發起者,獨立於zookeeper叢集的角色。
02 / ZooKeeper讀寫機制
在ZooKeeper的選舉中,如果過半的節點都選一個節點為leader的話,那麼這個節點就會是leader節點,也就是因為這個原因,ZooKeeper叢集,只要有過半的節點是存活的,那麼這個ZooKeeper就可以正常的提供服務。比如有5個ZooKeeper節點,其中有2個節點當機了,這個時候還有3個節點存活,存活個數超過半數,此時叢集還是正常提供服務,所以ZooKeeper叢集本生是沒有高可用問題的。又因為存活的判斷依據是超過半數,所以我們一般搭建ZooKeeper叢集的時候,都使用奇數臺,這樣會比較節約機器,比如我們安裝一個6臺的ZooKeeper叢集,如果當機了3臺就會導致叢集不可用,因為這個時候存活的節點數沒有超過半數了,所以6臺和5臺的效果是一樣的,我們用5臺比較合適。
對應一個ZooKeeper叢集,我們可能有多個客戶端,客戶端能任意連線其中一臺ZooKeeper節點,但是所有的客戶端都只能往leader節點上面去寫資料,所有的客戶端能從所有的節點上面讀取資料。如果有客戶端連線的是follower節點,然後往follower上傳送了寫資料的請求,這個時候follower就會把這個寫請求轉發給leader節點處理。leader接受到寫請求就會往其他節點(包括自己)同步資料,如果過半的節點接受到訊息後傳送回來ack訊息,那麼leader節點就對這條訊息進行commit,commit後該訊息就對使用者可見了。因為需要過半的節點傳送ack後,leader才對訊息進行commit,這個時候會有一個問題,如果叢集越大,那麼等待過半節點傳送回來ack訊息這個過程就需要越久,也就是說節點越多雖然會增加叢集的讀效能,但是會影響到叢集的寫效能,所以我們一般建議ZooKeeper的叢集規模在3到5個節點左右。為了解決這個問題,後來的ZooKeeper中增加了一個observer 的角色,這個節點不參與投票,只是負責同步資料。比如我們leader寫資料需要過半的節點傳送ack響應,這個observer節點是不參與過半的數量統計的。它只是負責從leader同步資料,然後提供給客戶端讀取,所以引入這個角色目的就是為了增加叢集讀的效能,然後不影響叢集的寫效能。使用者搭建叢集的時候可以自己設定該角色。
03 / Zookeeper 特點
0x01、一致性
client客戶端無論連線到叢集中的哪個節點,讀到的資料都是一樣的
0x02、實時性
ZooKeeper保證客戶端在一定的時間間隔內獲得結果,包括成功和失敗,但是由於網路延遲原因,ZooKeeper不能保證兩臺客戶端同時得到剛更新的訊息。如果都需要最新的訊息需要呼叫sync()介面。
0x03、原子性
leader在同步資料的時候,同步過程保證事務性,要麼都成功,要麼都失敗。
0x04、順序性
一臺伺服器上如果訊息a在訊息b前釋出,那麼所有的server上的訊息a都是在訊息b前釋出的。
04 / Zookeeper資料一致性保證
剛剛我們看到了ZooKeeper有多個特點,但是我相信多個特點中,大家最好奇都就是Zookeeper是如何保證資料一致性的。ZooKeeper保證資料一致性用的是ZAB協議。透過這個協議來進行ZooKeeper叢集間的資料同步,保證資料的一致性。
0x01、兩階段提交+過半寫機制
ZooKeeper寫資料的機制是客戶端把寫請求傳送到leader節點上(如果傳送的是follower節點,follower節點會把寫請求轉發到leader節點),leader節點會把資料透過proposal請求傳送到所有節點(包括自己),所有到節點接受到資料以後都會寫到自己到本地磁碟上面,寫好了以後會傳送一個ack請求給leader,leader只要接受到過半的節點傳送ack響應回來,就會傳送commit訊息給各個節點,各個節點就會把訊息放入到記憶體中(放記憶體是為了保證高效能),該訊息就會使用者可見了。那麼這個時候,如果ZooKeeper要想保證資料一致性,就需要考慮如下兩個情況,情況一:leader執行commit了,還沒來得及給follower傳送commit的時候,leader當機了,這個時候如何保證訊息一致性?情況二:客戶端把訊息寫到leader了,但是leader還沒傳送proposal訊息給其他節點,這個時候leader當機了,leader當機後恢復的時候此訊息又該如何處理?
0x02、ZAB的崩潰恢復機制
針對情況一,當leader當機以後,ZooKeeper會選舉出來新的leader,新的leader啟動以後要到磁碟上面去檢查是否存在沒有commit的訊息,如果存在,就繼續檢檢視其他follower有沒有對這條訊息進行了commit,如果有過半節點對這條訊息進行了ack,但是沒有commit,那麼新對leader要完成commit的操作。
0x03、ZAB恢復中刪除資料機制
針對情況二,客戶端把訊息寫到leader了,但是leader還沒傳送portal訊息給其他節點,這個時候leader當機了,這個時候對於使用者來說,這條訊息是寫失敗的。假設過了一段時間以後leader節點又恢復了,不過這個時候角色就變為了follower了,它在檢查自己磁碟的時候會發現自己有一條訊息沒有進行commit,此時就會檢測訊息的編號,訊息是有編號的,由高32位和低32位組成,高32位是用來體現是否發生過leader切換的,低32位就是展示訊息的順序的。這個時候當前的節點就會根據高32位知道目前leader已經切換過了,所以就把當前的訊息刪除,然後從新的leader同步資料,這樣保證了資料一致性。
最後
各位同學,ZooKeeper是我們學習架構的過程當中,非常非常重要的一個知識點,我們今天只是從其中一些角度去分析的ZooKeeper,我們後面會有很多文章從不同角度深入全面的剖析ZooKeeper,幫助大家掌握ZooKeeper,歡迎大家持續關注。
領取更多免費技術資料及影片
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69976011/viewspace-2697197/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- Zookeeper基礎原理&應用場景詳解
- Zookeeper應用場景
- RabbitMQ核心元件及應用場景MQ元件
- 【分散式】Zookeeper應用場景分散式
- zookeeper使用(四)--應用場景
- ZooKeeper分散式專題(二) -- zookeeper應用場景及資料模型分散式模型
- Zookeeper應用場景和ZAB協議協議
- Tornado原理淺析及應用場景探討
- Hive簡介、應用場景及架構原理Hive架構
- 海外IP池的工作原理及應用場景
- Zookeeper應用場景彙總(超詳細)
- Mybatis-Plus的應用場景及注入SQL原理分析MyBatisSQL
- 堆排序原理及其應用場景排序
- HDFS應用場景、原理、基本架構及使用方法概述架構
- redis應用場景及例項Redis
- Webpack 下使用 web workers 及 基本原理 和 應用場景Web
- Flink基本原理與應用場景
- 圖資料庫及應用場景資料庫
- 深入瞭解Zookeeper核心原理
- Redis的應用場景及優缺點Redis
- Fiddler(一)Fiddler介紹及應用場景
- Pytest的斷言方式及應用場景
- Redis應用場景及快取問題Redis快取
- HTAP資料庫及應用場景分析資料庫
- redis資料型別及應用場景Redis資料型別
- Java反射詳解:入門+使用+原理+應用場景Java反射
- Redis的資料結構及應用場景Redis資料結構
- zookeeper-操作與應用場景-《每日五分鐘搞定大資料》大資料
- 好程式設計師Java教程分享Zookeeper基本原理與運用場景程式設計師Java
- 3.4 應用場景
- DDD應用場景
- ES 應用場景
- snapshot應用場景
- jeesz-zookeeper使用場景
- IO多路複用原理&場景
- 分散式資料庫核心系統應用場景和需求分散式資料庫
- RabbitMQ 的應用場景以及基本原理介紹MQ
- RabbitMQ的應用場景以及基本原理介紹MQ