從Paxos到Zookeeper分布式一致性原理與實踐(博文視點出品)

2021-04-25 20:02 更新

從Paxos到Zookeeper分布式一致性原理與實踐(博文視點出品)

倪超 著

  • 出版社: 電子工業(yè)出版社
  • ISBN:9787121249679
  • 版次:1
  • 商品編碼:11622772
  • 品牌:博文視點
  • 包裝:平裝
  • 開本:16開
  • 出版時間:2015-02-01
  • 用紙:膠版紙
  • 頁數(shù):420
  • 字數(shù):548000
  • 正文語種:中文

點此購買


編輯推薦

適讀人群 :大型網(wǎng)站和分布式系統(tǒng)的開發(fā)人員,對分布式一致性和ZooKeeper感興趣的IT從業(yè)人員。

國內(nèi)罕見系統(tǒng)講解ZooKeeper這一應(yīng)用廣泛、成熟的分布式協(xié)調(diào)框架之技術(shù)書。

 原理深入,闡述清晰,覆蓋ACID、CAP、BASE,二階段/三階段提交,Paxos、ZAB協(xié)議等熱門話題。

 徹底剖析分布式一致性問題,并給出相應(yīng)系統(tǒng)思路,以及完整解決方案及實戰(zhàn)參考。

 無論開發(fā)人員,還是運維人士,都可通過書中ZooKeeper使用方法、內(nèi)部實現(xiàn)及運維技巧來全面提升。

內(nèi)容簡介

  《Paxos到Zookeeper 分布式一致性原理與實踐》從分布式一致性的理論出發(fā),向讀者簡要介紹幾種典型的分布式一致性協(xié)議,以及解決分布式一致性問題的思路,其中重點講解了Paxos和ZAB協(xié)議。同時,本書深入介紹了分布式一致性問題的工業(yè)解決方案——ZooKeeper,并著重向讀者展示這一分布式協(xié)調(diào)框架的使用方法、內(nèi)部實現(xiàn)及運維技巧,旨在幫助讀者全面了解ZooKeeper,并更好地使用和運維ZooKeeper。全書共8章,分為五部分:前一部分(第1章)主要介紹了計算機系統(tǒng)從集中式向分布式系統(tǒng)演變過程中面臨的挑戰(zhàn),并簡要介紹了ACID、CAP和BASE等經(jīng)典分布式理論;第二部分(第2~4章)介紹了2PC、3PC和Paxos三種分布式一致性協(xié)議,并著重講解了ZooKeeper中使用的一致性協(xié)議——ZAB協(xié)議;第三部分(第5~6章)介紹了ZooKeeper的使用方法,包括客戶端API的使用以及對ZooKeeper服務(wù)的部署與運行,并結(jié)合真實的分布式應(yīng)用場景,總結(jié)了ZooKeeper使用實踐;第四部分(第7章)對ZooKeeper的架構(gòu)設(shè)計和實現(xiàn)原理進行了深入分析,包含系統(tǒng)模型、Leader選舉、客戶端與服務(wù)端的工作原理、請求處理,以及服務(wù)器角色的工作流程和數(shù)據(jù)存儲等;第五部分(第8章)介紹了ZooKeeper的運維實踐,包括配置詳解和監(jiān)控管理等,重點講解了如何構(gòu)建一個高可用的ZooKeeper服務(wù)。


作者簡介

  倪超,阿里巴巴集團高級研發(fā)工程師,國家認證系統(tǒng)分析師,畢業(yè)于杭州電子科技大學(xué)計算機系。2010年加入阿里巴巴中間件團隊擔(dān)任研發(fā)實習(xí)崗位,一直從事ZooKeeper的開發(fā)與運維工作,從中學(xué)習(xí)與總結(jié)了不少分布式一致性相關(guān)的理論與實踐經(jīng)驗,尤其對ZooKeeper及其相關(guān)技術(shù)有非常深入的研究。目前在中間件團隊專家組任職產(chǎn)品經(jīng)理,負責(zé)分布式產(chǎn)品的產(chǎn)品化和云計算化改造工作。


精彩書評

  ★感謝軟件開源和知識開源,新浪愛彩利用各開源軟件和算法,構(gòu)建了核心交易系統(tǒng)和分布式中間件系統(tǒng):利用ZooKeeper 構(gòu)建了分布式 ID 生成器、分布式單例控制器、Dubbo RPC 框架,以及基于 Hadoop/JStorm/Spark 體系的業(yè)務(wù)系統(tǒng),等等。ZooKeeper 的穩(wěn)定性和對一致性的保證一直為業(yè)界所稱道,在大量的分布式系統(tǒng)和開源組件中得到應(yīng)用。本書是作者在長期使用 ZooKeeper 后深入研究其算法原理和源代碼的總結(jié),將對讀者在分布式一致性的理論學(xué)習(xí)與實踐上有啟發(fā)意義。

  ——新浪愛彩首席架構(gòu)師 周鋒

    ★分布式一致性是中國銀聯(lián)風(fēng)控系統(tǒng)架構(gòu)與設(shè)計的重要目標(biāo),新一代的銀聯(lián)反洗錢交易實時分析系統(tǒng)采用 Storm 進行大數(shù)據(jù)的實時計算,ZooKeeper 作為 Storm 的重要組成部分,為數(shù)據(jù)一致性提供了關(guān)鍵保障。本書深入淺出地描述了分布式一致性這一問題的由來,并對 ZooKeeper 在 Storm、Hadoop 和 HBase 等大型分布式系統(tǒng)中的應(yīng)用場景進行了詳盡介紹,針對 ZooKeeper在分布式系統(tǒng)中的業(yè)務(wù)實踐與運維保障提供了重要參考。

  ——中國銀聯(lián)反洗錢系統(tǒng)核心負責(zé)人 羅科勤

    ★分布式地理信息系統(tǒng)的研發(fā)挑戰(zhàn)主要在于它的地理信息共享和分布式協(xié)調(diào)操作,ZooKeeper 作為一個針對大型分布式系統(tǒng)的高可靠協(xié)調(diào)系統(tǒng),提供的功能包括:配置維護、名字服務(wù)、分布式同步和組服務(wù)等,正好能夠解決地信系統(tǒng)中的諸多分布式一致性問題。該書兼顧分布式一致性的理論和實踐,并重點講解了 ZooKeeper,適合不同層次的讀者閱讀。

  ——浙江省測繪局地信系統(tǒng)設(shè)計師 王浩烽

  ★騰訊在 2010 年啟動建設(shè)開放云平臺時,面臨著海量第三方虛擬機之間訪問限制規(guī)則以及內(nèi)網(wǎng)透明負載均衡配置的管理等問題。引入 ZooKeeper 之后,一直穩(wěn)定運行至今,利用其發(fā)布訂閱特性很好地保證了規(guī)則數(shù)據(jù)和配置信息的一致性,確保了服務(wù)的可用性。本書從分布式一致性理論出發(fā),再以ZooKeeper 系統(tǒng)為例詳盡地介紹了這個開源系統(tǒng)的架構(gòu)與實現(xiàn),并結(jié)合實際的應(yīng)用場景和運維經(jīng)驗為在實戰(zhàn)中面臨分布式問題的讀者提供了重要參考。

  ——騰訊企業(yè)級產(chǎn)品中心架構(gòu)師 陳盛龍

    ★一致性是計算機學(xué)科中“硬”和重要的問題之一,可見寫這樣一個主題挑戰(zhàn)之大。阿里巴巴業(yè)務(wù)龐大,倪超之前維護的為整個集團提供一致性方案的 ZooKeeper 集群,場景之復(fù)雜、規(guī)模之大在國內(nèi)甚至世界上都可能是罕見的。本人由于工作需要對 Paxos 和 ZooKeeper 進行了粗淺的學(xué)習(xí),所以有機會和倪超有過這方面的交流,樂自不言,獲益彼多。本書兼顧理論與實踐,希望讓讀者讀完之后有所提升:使用上知其所以然,架構(gòu)上能選擇出合適又低成本的方案。

  ——阿里巴巴 Dubbo 框架、PaaS 平臺資深架構(gòu)師 & 核心開發(fā) 李鼎

    ★在我的工作經(jīng)歷中,有多次與分布式系統(tǒng)的配置管理中心打過交道,比如之前在老東家阿里巴巴負責(zé) HSF 服務(wù)框架,以及最近在陌陌負責(zé)的 MOA 服務(wù)框架的工作?;诤唵慰捎玫脑瓌t,這些場景都沒有選擇使用 ZooKeeper,而是自己實現(xiàn)了配置管理系統(tǒng)。但最近在參與分布式緩存服務(wù)建設(shè)的過程中,我們發(fā)現(xiàn)已經(jīng)無法再繞開分布式協(xié)調(diào)問題,這時,ZooKeeper作為行業(yè)的成熟實踐就成了我們的優(yōu)選。這本書的作者倪超是我在阿里的同事,一直從事著與 ZooKeeper 相關(guān)的工作 在這個領(lǐng)域積累了豐富的經(jīng)驗。本書從理論、設(shè)計實現(xiàn)和應(yīng)用場景等多個方面對 ZooKeeper 進行了深入介紹,非常值得一讀。

  ——陌陌基礎(chǔ)平臺部主管 宓學(xué)強

    ★搜狐從 2009 年微博時代初期就利用 ZooKeeper 的發(fā)布與訂閱模型實現(xiàn)了對 CDN URL 和一些基本管理配置的動態(tài)加載。至今 ZooKeeper 已經(jīng)被運用在了搜狐各大業(yè)務(wù)線上,完成了許多分布式高可用服務(wù)的構(gòu)建,范圍涉及分布式緩存、服務(wù)化框架和前端業(yè)務(wù)系統(tǒng)等等,幫助團隊解決了分布式方面的主要技術(shù)障礙,大大提高了業(yè)務(wù)穩(wěn)定性和運維效率。本書全面詳盡地介紹了分布式環(huán)境中各個典型場景下的 ZooKeeper 應(yīng)用實例,為讀者構(gòu)建自己的分布式高可用服務(wù)提供了參考。

  ——搜狐移動事業(yè)部高級運維主管 劉鵬


目錄


第1章 分布式架構(gòu)
1.1 從集中式到分布式
1.1.1 集中式的特點
1.1.2 分布式的特點
1.1.3 分布式環(huán)境的各種問題
1.2 從ACID到CAP/BASE
1.2.1 ACID
1.2.2 分布式事務(wù)
1.2.3 CAP和BASE理論
小結(jié)

第2章 一致性協(xié)議
2.1 2PC與3PC
2.1.1 2PC
2.1.2 3PC
2.2 Paxos算法
2.2.1 追本溯源
2.2.2 Paxos理論的誕生
2.2.3 Paxos算法詳解
小結(jié)

第3章 Paxos的工程實踐
3.1 Chubby
3.1.1 概述
3.1.2 應(yīng)用場景
3.1.3 設(shè)計目標(biāo)
3.1.4 Chubby技術(shù)架構(gòu)
3.1.5 Paxos協(xié)議實現(xiàn)
3.2 Hypertable
3.2.1 概述
3.2.2 算法實現(xiàn)
小結(jié)

第4章 ZooKeeper與Paxos
4.1 初識ZooKeeper
4.1.1 ZooKeeper介紹
4.1.2 ZooKeeper從何而來
4.1.3 ZooKeeper的基本概念
4.1.4 為什么選擇ZooKeeper
4.2 ZooKeeper的ZAB協(xié)議
4.2.1 ZAB協(xié)議
4.2.2 協(xié)議介紹
4.2.3 深入ZAB協(xié)議
4.2.4 ZAB與Paxos算法的聯(lián)系與區(qū)別
小結(jié)

第5章 使用ZooKeeper
5.1 部署與運行
5.1.1 系統(tǒng)環(huán)境
5.1.2 集群與單機
5.1.3 運行服務(wù)
5.2 客戶端腳本
5.2.1 創(chuàng)建
5.2.2 讀取
5.2.3 更新
5.2.4 刪除
5.3 Java客戶端API使用
5.3.1 創(chuàng)建會話
5.3.2 創(chuàng)建節(jié)點
5.3.3 刪除節(jié)點
5.3.4 讀取數(shù)據(jù)
5.3.5 更新數(shù)據(jù)
5.3.6 檢測節(jié)點是否存在
5.3.7 權(quán)限控制
5.4 開源客戶端
5.4.1 ZkClient
5.4.2 Curator
小結(jié)

第6章 ZooKeeper的典型應(yīng)用場景
6.1 典型應(yīng)用場景及實現(xiàn)注
6.1.1 數(shù)據(jù)發(fā)布/訂閱
6.1.2 負載均衡
6.1.3 命名服務(wù)
6.1.4 分布式協(xié)調(diào)/通知
6.1.5 集群管理
6.1.6 Master選舉
6.1.7 分布式鎖
6.1.8 分布式隊列
小結(jié)
6.2 ZooKeeper在大型分布式系統(tǒng)中的應(yīng)用
6.2.1 Hadoop
6.2.2 HBase
6.2.3 Kafka
6.3 ZooKeeper在阿里巴巴的實踐與應(yīng)用
6.3.1 案例一 消息中間件:Metamorphosis
6.3.2 案例二 RPC服務(wù)框架:Dubbo
6.3.3 案例三 基于MySQL Binlog的增量訂閱和消費組件:Canal
6.3.4 案例四 分布式數(shù)據(jù)庫同步系統(tǒng):Otter
6.3.5 案例五 輕量級分布式通用搜索平臺:終搜
6.3.6 案例六 實時計算引擎:JStorm
小結(jié)

第7章 ZooKeeper技術(shù)內(nèi)幕
7.1 系統(tǒng)模型
7.1.1 數(shù)據(jù)模型
7.1.2 節(jié)點特性
7.1.3 版本――保證分布式數(shù)據(jù)原子性操作
7.1.4 Watcher――數(shù)據(jù)變更的通知
7.1.5 ACL――保障數(shù)據(jù)的安全
7.2 序列化與協(xié)議
7.2.1 Jute介紹
7.2.2 使用Jute進行序列化
7.2.3 深入Jute
7.2.4 通信協(xié)議
7.3 客戶端
7.3.1 一次會話的創(chuàng)建過程
7.3.2 服務(wù)器地址列表
7.3.3 ClientCnxn:網(wǎng)絡(luò)I/O
7.4 會話
7.4.1 會話狀態(tài)
7.4.2 會話創(chuàng)建
7.4.3 會話管理
7.4.4 會話清理
7.4.5 重連
7.5 服務(wù)器啟動
7.5.1 單機版服務(wù)器啟動
7.5.2 集群版服務(wù)器啟動
7.6 Leader選舉
7.6.1 Leader選舉概述
7.6.2 Leader選舉的算法分析
7.6.3 Leader選舉的實現(xiàn)細節(jié)
7.7 各服務(wù)器角色介紹
7.7.1 Leader
7.7.2 Follower
7.7.3 Observer
7.7.4 集群間消息通信
7.8 請求處理
7.8.1 會話創(chuàng)建請求
7.8.2 SetData請求
7.8.3 事務(wù)請求轉(zhuǎn)發(fā)
7.8.4 GetData請求
7.9 數(shù)據(jù)與存儲
7.9.1 內(nèi)存數(shù)據(jù)
7.9.2 事務(wù)日志
7.9.3 snapshot――數(shù)據(jù)快照
7.9.4 初始化
7.9.5 數(shù)據(jù)同步
小結(jié)

第8章 ZooKeeper運維
8.1 配置詳解
8.1.1 基本配置
8.1.2 高級配置
8.2 四字命令
8.3 JMX
8.3.1 開啟遠程JMX
8.3.2 通過JConsole連接ZooKeeper
8.4 監(jiān)控
8.4.1 實時監(jiān)控
8.4.2 數(shù)據(jù)統(tǒng)計
8.5 構(gòu)建一個高可用的集群
8.5.1 集群組成
8.5.2 容災(zāi)
8.5.3 擴容與縮容
8.6 日常運維
8.6.1 數(shù)據(jù)與日志管理
8.6.2 Too many connections
8.6.3 磁盤管理
小結(jié)
附錄A Windows平臺上部署ZooKeeper
附錄B 從源代碼開始構(gòu)建
附錄C 各發(fā)行版本重大更新記錄
附錄D ZooKeeper源代碼閱讀指引


精彩書摘

  第1章

  分布式架構(gòu)

  隨著計算機系統(tǒng)規(guī)模變得越來越大,將所有的業(yè)務(wù)單元集中部署在一個或若干個大型機上的體系結(jié)構(gòu),已經(jīng)越來越不能滿足當(dāng)今計算機系統(tǒng),尤其是大型互聯(lián)網(wǎng)系統(tǒng)的快速發(fā)展,各種靈活多變的系統(tǒng)架構(gòu)模型層出不窮。同時,隨著微型計算機的出現(xiàn),越來越多廉價的PC機成為了各大企業(yè)IT架構(gòu)的首選,分布式的處理方式越來越受到業(yè)界的青睞——計算機系統(tǒng)正在經(jīng)歷一場前所未有的從集中式向分布式架構(gòu)的變革。

  1.1 從集中式到分布式

  自20世紀60年代大型主機被發(fā)明出來以后,憑借其超強的計算和I/O處理能力以及在穩(wěn)定性和安全性方面的卓越表現(xiàn),在很長一段時間內(nèi),大型主機引領(lǐng)了計算機行業(yè)以及商業(yè)計算領(lǐng)域的發(fā)展。在大型主機的研發(fā)上最知名的當(dāng)屬IBM,其主導(dǎo)研發(fā)的革命性產(chǎn)品System/360系列大型主機,是計算機發(fā)展史上的一個里程碑,與波音707和福特T型車齊名,被譽為20世紀最重要的三大商業(yè)成就,并一度成為了大型主機的代名詞。從那時起,IT界進入了大型主機時代。

  伴隨著大型主機時代的到來,集中式的計算機系統(tǒng)架構(gòu)也成為了主流。在那個時候,由于大型主機卓越的性能和良好的穩(wěn)定性,其在單機處理能力方面的優(yōu)勢非常明顯,使得IT系統(tǒng)快速進入了集中式處理階段,其對應(yīng)的計算機系統(tǒng)稱為集中式系統(tǒng)。但從20世紀80年代以來,計算機系統(tǒng)向網(wǎng)絡(luò)化和微型化的發(fā)展日趨明顯,傳統(tǒng)的集中式處理模式越來越不能適應(yīng)人們的需求。

  首先,大型主機的人才培養(yǎng)成本非常之高。通常一臺大型主機匯集了大量精密的計算機組件,操作非常復(fù)雜,這對一個運維人員掌握其技術(shù)細節(jié)提出了非常高的要求。

  其次,大型主機也是非常昂貴的。通常一臺配置較好的IBM大型主機,其售價可能在上百萬美元甚至更高,因此也只有像政府、金融和電信等企業(yè)才有能力采購大型主機。

  另外,集中式系統(tǒng)具有明顯的單點問題。大型主機雖然在性能和穩(wěn)定性方面表現(xiàn)卓越,但這并不代表其永遠不會出現(xiàn)故障。一旦一臺大型主機出現(xiàn)了故障,那么整個系統(tǒng)將處于不可用狀態(tài),其后果相當(dāng)嚴重。最后,隨著業(yè)務(wù)的不斷發(fā)展,用戶訪問量迅速提高,計算機系統(tǒng)的規(guī)模也在不斷擴大,在單一大型主機上進行系統(tǒng)的擴容往往比較困難。

  而另一方面,隨著PC機性能的不斷提升和網(wǎng)絡(luò)技術(shù)的快速普及,大型主機的市場份額變得越來越小,很多企業(yè)開始放棄原來的大型主機,而改用小型機和普通PC服務(wù)器來搭建分布式的計算機系統(tǒng)。

  其中最為典型的就是阿里巴巴集團的“去IOE”運動。從2008年開始,阿里巴巴的各項業(yè)務(wù)都進入了井噴式的發(fā)展階段,這對于后臺IT系統(tǒng)的計算與存儲能力提出了非常高的要求,一味地針對小型機和高端存儲進行不斷擴容,無疑會產(chǎn)生巨大的成本。同時,集中式的系統(tǒng)架構(gòu)體系也存在諸多單點問題,完全無法滿足互聯(lián)網(wǎng)應(yīng)用爆炸式的發(fā)展需求。因此,為了解決業(yè)務(wù)快速發(fā)展給IT系統(tǒng)帶來的巨大挑戰(zhàn),從2009年開始,阿里集團啟動了“去IOE”計劃,其電商系統(tǒng)開始正式邁入分布式系統(tǒng)時代。

  1.1.1 集中式的特點

  所謂的集中式系統(tǒng)就是指由一臺或多臺主計算機組成中心節(jié)點,數(shù)據(jù)集中存儲于這個中心節(jié)點中,并且整個系統(tǒng)的所有業(yè)務(wù)單元都集中部署在這個中心節(jié)點上,系統(tǒng)的所有功能均由其集中處理。也就是說,在集中式系統(tǒng)中,每個終端或客戶端機器僅僅負責(zé)數(shù)據(jù)的錄入和輸出,而數(shù)據(jù)的存儲與控制處理完全交由主機來完成。

  集中式系統(tǒng)最大的特點就是部署結(jié)構(gòu)簡單。由于集中式系統(tǒng)往往基于底層性能卓越的大型主機,因此無須考慮如何對服務(wù)進行多個節(jié)點的部署,也就不用考慮多個節(jié)點之間的分布式協(xié)作問題。

  1.1.2 分布式的特點

  在《分布式系統(tǒng)概念與設(shè)計》注 一書中,對分布式系統(tǒng)做了如下定義:

  分布式系統(tǒng)是一個硬件或軟件組件分布在不同的網(wǎng)絡(luò)計算機上,彼此之間僅僅通過消息傳遞進行通信和協(xié)調(diào)的系統(tǒng)。

  上面這個簡單的定義涵蓋了幾乎所有有效地部署了網(wǎng)絡(luò)化計算機的系統(tǒng)。嚴格地講,同一個分布式系統(tǒng)中的計算機在空間部署上是可以隨意分布的,這些計算機可能被放在不同的機柜上,也可能在不同的機房中,甚至分布在不同的城市。無論如何,一個標(biāo)準的分布式系統(tǒng)在沒有任何特定業(yè)務(wù)邏輯約束的情況下,都會有如下幾個特征。

  分布性

  分布式系統(tǒng)中的多臺計算機都會在空間上隨意分布,同時,機器的分布情況也會隨時變動。

  對等性

  分布式系統(tǒng)中的計算機沒有主/從之分,既沒有控制整個系統(tǒng)的主機,也沒有被控制的從機,組成分布式系統(tǒng)的所有計算機節(jié)點都是對等的。副本(Replica)是分布式系統(tǒng)最常見的概念之一,指的是分布式系統(tǒng)對數(shù)據(jù)和服務(wù)提供的一種冗余方式。在常見的分布式系統(tǒng)中,為了對外提供高可用的服務(wù),我們往往會對數(shù)據(jù)和服務(wù)進行副本處理。數(shù)據(jù)副本是指在不同的節(jié)點上持久化同一份數(shù)據(jù),當(dāng)某一個節(jié)點上存儲的數(shù)據(jù)丟失時,可以從副本上讀取到該數(shù)據(jù),這是解決分布式系統(tǒng)數(shù)據(jù)丟失問題最為有效的手段。另一類副本是服務(wù)副本,指多個節(jié)點提供同樣的服務(wù),每個節(jié)點都有能力接收來自外部的請求并進行相應(yīng)的處理。

  并發(fā)性

  在“問題的提出”部分,我們已經(jīng)提到過與“更新的并發(fā)性”相關(guān)的內(nèi)容。在一個計算機網(wǎng)絡(luò)中,程序運行過程中的并發(fā)性操作是非常常見的行為,例如同一個分布式系統(tǒng)中的多個節(jié)點,可能會并發(fā)地操作一些共享的資源,諸如數(shù)據(jù)庫或分布式存儲等,如何準確并高效地協(xié)調(diào)分布式并發(fā)操作也成為了分布式系統(tǒng)架構(gòu)與設(shè)計中最大的挑戰(zhàn)之一?! ∪狈θ謺r鐘  在上面的講解中,我們已經(jīng)了解到,一個典型的分布式系統(tǒng)是由一系列在空間上隨意分布的多個進程組成的,具有明顯的分布性,這些進程之間通過交換消息來進行相互通信。因此,在分布式系統(tǒng)中,很難定義兩個事件究竟誰先誰后,原因就是因為分布式系統(tǒng)缺乏一個全局的時鐘序列控制。關(guān)于分布式系統(tǒng)的時鐘和事件順序,在Leslie Lamport注 的經(jīng)典論文Time, Clocks, and the Ordering of Events in a Distributed System注 中已經(jīng)做了非常深刻的講解。

  故障總是會發(fā)生

  組成分布式系統(tǒng)的所有計算機,都有可能發(fā)生任何形式的故障。一個被大量工程實踐所檢驗過的黃金定理是:任何在設(shè)計階段考慮到的異常情況,一定會在系統(tǒng)實際運行中發(fā)生,并且,在系統(tǒng)實際運行過程中還會遇到很多在設(shè)計時未能考慮到的異常故障。所以,除非需求指標(biāo)允許,在系統(tǒng)設(shè)計時不能放過任何異常情況。

  1.1.3 分布式環(huán)境的各種問題

  分布式系統(tǒng)體系結(jié)構(gòu)從其出現(xiàn)之初就伴隨著諸多的難題和挑戰(zhàn),本節(jié)將向讀者簡要的介紹分布式環(huán)境中一些典型的問題。

  通信異常

  從集中式向分布式演變的過程中,必然引入了網(wǎng)絡(luò)因素,而由于網(wǎng)絡(luò)本身的不可靠性,因此也引入了額外的問題。分布式系統(tǒng)需要在各個節(jié)點之間進行網(wǎng)絡(luò)通信,因此每次網(wǎng)絡(luò)通信都會伴隨著網(wǎng)絡(luò)不可用的風(fēng)險,網(wǎng)絡(luò)光纖、路由器或是DNS等硬件設(shè)備或是系統(tǒng)不可用都會導(dǎo)致最終分布式系統(tǒng)無法順利完成一次網(wǎng)絡(luò)通信。另外,即使分布式系統(tǒng)各節(jié)點之間的網(wǎng)絡(luò)通信能夠正常進行,其延時也會遠大于單機操作。通常我們認為在現(xiàn)代計算機體系結(jié)構(gòu)中,單機內(nèi)存訪問的延時在納秒數(shù)量級(通常是10ns左右),而正常的一次網(wǎng)絡(luò)通信的延遲在0.1~1ms左右(相當(dāng)于內(nèi)存訪問延時的105~106倍),如此巨大的延時差別,也會影響消息的收發(fā)的過程,因此消息丟失和消息延遲變得非常普遍?! 【W(wǎng)絡(luò)分區(qū)  當(dāng)網(wǎng)絡(luò)由于發(fā)生異常情況,導(dǎo)致分布式系統(tǒng)中部分節(jié)點之間的網(wǎng)絡(luò)延時不斷增大,最終導(dǎo)致組成分布式系統(tǒng)的所有節(jié)點中,只有部分節(jié)點之間能夠進行正常通信,而另一些節(jié)點則不能——我們將這個現(xiàn)象稱為網(wǎng)絡(luò)分區(qū),就是俗稱的“腦裂”。當(dāng)網(wǎng)絡(luò)分區(qū)出現(xiàn)時,分布式系統(tǒng)會出現(xiàn)局部小集群,在極端情況下,這些局部小集群會獨立完成原本需要整個分布式系統(tǒng)才能完成的功能,包括對數(shù)據(jù)的事務(wù)處理,這就對分布式一致性提出了非常大的挑戰(zhàn)。

  三態(tài)

  從上面的介紹中,我們已經(jīng)了解到了在分布式環(huán)境下,網(wǎng)絡(luò)可能會出現(xiàn)各式各樣的問題,因此分布式系統(tǒng)的每一次請求與響應(yīng),存在特有的“三態(tài)”概念,即成功、失敗與超時。在傳統(tǒng)的單機系統(tǒng)中,應(yīng)用程序在調(diào)用一個函數(shù)之后,能夠得到一個非常明確的響應(yīng):成功或失敗。而在分布式系統(tǒng)中,由于網(wǎng)絡(luò)是不可靠的,雖然在絕大部分情況下,網(wǎng)絡(luò)通信也能夠接收到成功或失敗的響應(yīng),但是當(dāng)網(wǎng)絡(luò)出現(xiàn)異常的情況下,就可能會出現(xiàn)超時現(xiàn)象,通常有以下兩種情況:  由于網(wǎng)絡(luò)原因,該請求(消息)并沒有被成功地發(fā)送到接收方,而是在發(fā)送過程就發(fā)生了消息丟失現(xiàn)象。

  該請求(消息)成功的被接收方接收后,并進行了處理,但是在將響應(yīng)反饋給發(fā)送方的過程中,發(fā)生了消息丟失現(xiàn)象?! ‘?dāng)出現(xiàn)這樣的超時現(xiàn)象時,網(wǎng)絡(luò)通信的發(fā)起方是無法確定當(dāng)前請求是否被成功處理的。

  節(jié)點故障

  節(jié)點故障則是分布式環(huán)境下另一個比較常見的問題,指的是組成分布式系統(tǒng)的服務(wù)器節(jié)點出現(xiàn)的宕機或“僵死”現(xiàn)象。通常根據(jù)經(jīng)驗來說,每個節(jié)點都有可能會出現(xiàn)故障,并且每天都在發(fā)生。

  1.2 從ACID到CAP/BASE

  在上文中,我們講解了集中式系統(tǒng)和分布式系統(tǒng)各自的特點,同時也看到了在從集中式系統(tǒng)架構(gòu)向分布式系統(tǒng)架構(gòu)變遷的過程中會碰到的一系列問題。接下來,我們再重點看看在分布式系統(tǒng)事務(wù)處理與數(shù)據(jù)一致性上遇到的種種挑戰(zhàn)。

  1.2.1 ACID

  事務(wù)(Transaction)是由一系列對系統(tǒng)中數(shù)據(jù)進行訪問與更新的操作所組成的一個程序執(zhí)行邏輯單元(Unit),狹義上的事務(wù)特指數(shù)據(jù)庫事務(wù)。一方面,當(dāng)多個應(yīng)用程序并發(fā)訪問數(shù)據(jù)庫時,事務(wù)可以在這些應(yīng)用程序之間提供一個隔離方法,以防止彼此的操作互相干擾。另一方面,事務(wù)為數(shù)據(jù)庫操作序列提供了一個從失敗中恢復(fù)到正常狀態(tài)的方法,同時提供了數(shù)據(jù)庫即使在異常狀態(tài)下仍能保持數(shù)據(jù)一致性的方法。

  事務(wù)具有四個特征,分別是原子性(Atomicity)、一致性(Consistency)、隔離性(Isolation)和持久性(Durability),簡稱為事務(wù)的ACID特性。  原子性  事務(wù)的原子性是指事務(wù)必須是一個原子的操作序列單元。事務(wù)中包含的各項操作在一次執(zhí)行過程中,只允許出現(xiàn)以下兩種狀態(tài)之一。

  全部成功執(zhí)行。

  全部不執(zhí)行。

  任何一項操作失敗都將導(dǎo)致整個事務(wù)失敗,同時其他已經(jīng)被執(zhí)行的操作都將被撤銷并回滾,只有所有的操作全部成功,整個事務(wù)才算是成功完成。

  一致性

  事務(wù)的一致性是指事務(wù)的執(zhí)行不能破壞數(shù)據(jù)庫數(shù)據(jù)的完整性和一致性,一個事務(wù)在執(zhí)行之前和執(zhí)行之后,數(shù)據(jù)庫都必須處于一致性狀態(tài)。也就是說,事務(wù)執(zhí)行的結(jié)果必須是使數(shù)據(jù)庫從一個一致性狀態(tài)轉(zhuǎn)變到另一個一致性狀態(tài),因此當(dāng)數(shù)據(jù)庫只包含成功事務(wù)提交的結(jié)果時,就能說數(shù)據(jù)庫處于一致性狀態(tài)。而如果數(shù)據(jù)庫系統(tǒng)在運行過程中發(fā)生故障,有些事務(wù)尚未完成就被迫中斷,這些未完成的事務(wù)對數(shù)據(jù)庫所做的修改有一部分已寫入物理數(shù)據(jù)庫,這時數(shù)據(jù)庫就處于一種不正確的狀態(tài),或者說是不一致的狀態(tài)。

  隔離性

  事務(wù)的隔離性是指在并發(fā)環(huán)境中,并發(fā)的事務(wù)是相互隔離的,一個事務(wù)的執(zhí)行不能被其他事務(wù)干擾。也就是說,不同的事務(wù)并發(fā)操縱相同的數(shù)據(jù)時,每個事務(wù)都有各自完整的數(shù)據(jù)空間,即一個事務(wù)內(nèi)部的操作及使用的數(shù)據(jù)對其他并發(fā)事務(wù)是隔離的,并發(fā)執(zhí)行的各個事務(wù)之間不能互相干擾。

  在標(biāo)準SQL規(guī)范中,定義了4個事務(wù)隔離級別,不同的隔離級別對事務(wù)的處理不同,如未授權(quán)讀取、授權(quán)讀取、可重復(fù)讀取和串行化注 。

  未授權(quán)讀取

  未授權(quán)讀取也被稱為讀未提交(Read Uncommitted),該隔離級別允許臟讀取,其隔離級別最低。換句話說,如果一個事務(wù)正在處理某一數(shù)據(jù),并對其進行了更新,但同時尚未完成事務(wù),因此還沒有進行事務(wù)提交;而與此同時,允許另一個事務(wù)也能夠訪問該數(shù)據(jù)。舉個例子來說,事務(wù)A和事務(wù)B同時進行,事務(wù)A在整個執(zhí)行階段,會將某數(shù)據(jù)項的值從1開始,做一系列加法操作(比如說加1操作)直到變成10之后進行事務(wù)提交,此時,事務(wù)B能夠看到這個數(shù)據(jù)項在事務(wù)A操作過程中的所有中間值(如1變成2、2變成3等),而對這一系列的中間值的讀取就是未授權(quán)讀取。

  授權(quán)讀取

  授權(quán)讀取也被稱為讀已提交(Read Committed),它和未授權(quán)讀取非常相近,唯一的區(qū)別就是授權(quán)讀取只允許獲取已經(jīng)被提交的數(shù)據(jù)。同樣以上面的例子來說,事務(wù)A和事務(wù)B同時進行,事務(wù)A進行與上述同樣的操作,此時,事務(wù)B無法看到這個數(shù)據(jù)項在事務(wù)A操作過程中的所有中間值,只能看到最終的10。另外,如果說有一個事務(wù)C,和事務(wù)A進行非常類似的操作,只是事務(wù)C是將數(shù)據(jù)項從10加到20,此時事務(wù)B也同樣可以讀取到20,即授權(quán)讀取允許不可重復(fù)讀取。

  可重復(fù)讀取

  可重復(fù)讀?。≧epeatable Read),簡單地說,就是保證在事務(wù)處理過程中,多次讀取同一個數(shù)據(jù)時,其值都和事務(wù)開始時刻是一致的。因此該事務(wù)級別禁止了不可重復(fù)讀取和臟讀取,但是有可能出現(xiàn)幻影數(shù)據(jù)。所謂幻影數(shù)據(jù),就是指同樣的事務(wù)操作,在前后兩個時間段內(nèi)執(zhí)行對同一個數(shù)據(jù)項的讀取,可能出現(xiàn)不一致的結(jié)果。在上面的例子,可重復(fù)讀取隔離級別能夠保證事務(wù)B在第一次事務(wù)操作過程中,始終對數(shù)據(jù)項讀取到1,但是在下一次事務(wù)操作中,即使事務(wù)B(注意,事務(wù)名字雖然相同,但是指的是另一次事務(wù)操作)采用同樣的查詢方式,就可能會讀取到10或20。

  串行化

  串行化(Serializable)是最嚴格的事務(wù)隔離級別。它要求所有事務(wù)都被串行執(zhí)行,即事務(wù)只能一個接一個地進行處理,不能并發(fā)執(zhí)行。

  圖1-1展示了不同隔離級別下事務(wù)訪問數(shù)據(jù)的差異。

  圖1-1.4種隔離級別示意圖

  以上4個隔離級別的隔離性依次增強,分別解決不同的問題,表1-1對這4個隔離級別進行了一個簡單的對比。

  表1-1.隔離級別對比  隔離級別 臟讀 可重復(fù)讀 幻讀  未授權(quán)讀取 存在 不可以 存在

  授權(quán)讀取 不存在 不可以 存在

  可重復(fù)讀取 不存在 可以 存在

  串行化 不存在 可以 不存在

  事務(wù)隔離級別越高,就越能保證數(shù)據(jù)的完整性和一致性,但同時對并發(fā)性能的影響也越大。通常,對于絕大多數(shù)的應(yīng)用程序來說,可以優(yōu)先考慮將數(shù)據(jù)庫系統(tǒng)的隔離級別設(shè)置為授權(quán)讀取,這能夠在避免臟讀取的同時保證較好的并發(fā)性能。盡管這種事務(wù)隔離級別會導(dǎo)致不可重復(fù)讀、虛讀和第二類丟失更新等并發(fā)問題,但較為科學(xué)的做法是在可能出現(xiàn)這類問題的個別場合中,由應(yīng)用程序主動采用悲觀鎖或樂觀鎖來進行事務(wù)控制。

  持久性

  事務(wù)的持久性也被稱為永久性,是指一個事務(wù)一旦提交,它對數(shù)據(jù)庫中對應(yīng)數(shù)據(jù)的狀態(tài)變更就應(yīng)該是永久性的。換句話說,一旦某個事務(wù)成功結(jié)束,那么它對數(shù)據(jù)庫所做的更新就必須被永久保存下來——即使發(fā)生系統(tǒng)崩潰或機器宕機等故障,只要數(shù)據(jù)庫能夠重新啟動,那么一定能夠?qū)⑵浠謴?fù)到事務(wù)成功結(jié)束時的狀態(tài)。

  1.2.2 分布式事務(wù)

  隨著分布式計算的發(fā)展,事務(wù)在分布式計算領(lǐng)域中也得到了廣泛的應(yīng)用。在單機數(shù)據(jù)庫中,我們很容易能夠?qū)崿F(xiàn)一套滿足ACID特性的事務(wù)處理系統(tǒng),但在分布式數(shù)據(jù)庫中,數(shù)據(jù)分散在各臺不同的機器上,如何對這些數(shù)據(jù)進行分布式的事務(wù)處理具有非常大的挑戰(zhàn)。在1.1.3節(jié)中,我們已經(jīng)講解了分布式環(huán)境中會碰到的種種問題,其中就包括機器宕機和各種網(wǎng)絡(luò)異常等。盡管存在這種種分布式問題,但是在分布式計算領(lǐng)域,為了保證分布式應(yīng)用程序的可靠性,分布式事務(wù)是無法回避的。

  分布式事務(wù)是指事務(wù)的參與者、支持事務(wù)的服務(wù)器、資源服務(wù)器以及事務(wù)管理器分別位于分布式系統(tǒng)的不同節(jié)點之上。通常一個分布式事務(wù)中會涉及對多個數(shù)據(jù)源或業(yè)務(wù)系統(tǒng)的操作。

  我們可以設(shè)想一個最典型的分布式事務(wù)場景:一個跨銀行的轉(zhuǎn)賬操作涉及調(diào)用兩個異地的銀行服務(wù),其中一個是本地銀行提供的取款服務(wù),另一個則是目標(biāo)銀行提供的存款服務(wù),這兩個服務(wù)本身是無狀態(tài)并且是互相獨立的,共同構(gòu)成了一個完整的分布式事務(wù)。如果從本地銀行取款成功,但是因為某種原因存款服務(wù)失敗了,那么就必須回滾到取款前的狀態(tài),否則用戶可能會發(fā)現(xiàn)自己的錢不翼而飛了。

  從上面這個例子中,我們可以看到,一個分布式事務(wù)可以看作是由多個分布式的操作序列組成的,例如上面例子中的取款服務(wù)和存款服務(wù),通??梢园堰@一系列分布式的操作序列稱為子事務(wù)。因此,分布式事務(wù)也可以被定義為一種嵌套型的事務(wù),同時也就具有了ACID事務(wù)特性。但由于在分布式事務(wù)中,各個子事務(wù)的執(zhí)行是分布式的,因此要實現(xiàn)一種能夠保證ACID特性的分布式事務(wù)處理系統(tǒng)就顯得格外復(fù)雜。

  1.2.3 CAP和BASE理論

  對于本地事務(wù)處理或者是集中式的事務(wù)處理系統(tǒng),很顯然我們可以采用已經(jīng)被實踐證明很成熟的ACID模型來保證數(shù)據(jù)的嚴格一致性。而在1.2.2節(jié)中,我們也已經(jīng)看到,隨著分布式事務(wù)的出現(xiàn),傳統(tǒng)的單機事務(wù)模型已經(jīng)無法勝任。尤其是對于一個高訪問量、高并發(fā)的互聯(lián)網(wǎng)分布式系統(tǒng)來說,如果我們期望實現(xiàn)一套嚴格滿足ACID特性的分布式事務(wù),很可能出現(xiàn)的情況就是在系統(tǒng)的可用性和嚴格一致性之間出現(xiàn)沖突——因為當(dāng)我們要求分布式系統(tǒng)具有嚴格一致性時,很可能就需要犧牲掉系統(tǒng)的可用性。但毋庸置疑的一點是,可用性又是一個所有消費者不允許我們討價還價的系統(tǒng)屬性,比如說像淘寶網(wǎng)這樣的在線購物網(wǎng)站,就要求它能夠7?24小時不間斷地對外提供服務(wù),而對于一致性,則更加是所有消費者對于一個軟件系統(tǒng)的剛需。因此,在可用性和一致性之間永遠無法存在一個兩全其美的方案,于是如何構(gòu)建一個兼顧可用性和一致性的分布式系統(tǒng)成為了無數(shù)工程師探討的難題,出現(xiàn)了諸如CAP和BASE這樣的分布式系統(tǒng)經(jīng)典理論。

  CAP定理

  2000年7月,來自加州大學(xué)伯克利分校的Eric Brewer教授注 在ACM PODC(Principles of Distributed Computing)會議上,首次提出了著名的CAP猜想注 。2年后,來自麻省理工學(xué)院的Seth Gilbert和Nancy Lynch從理論上證明了Brewer教授CAP猜想的可行性注 ,從此,CAP理論正式在學(xué)術(shù)上成為了分布式計算領(lǐng)域的公認定理,并深深地影響了分布式計算的發(fā)展。

  CAP理論告訴我們,一個分布式系統(tǒng)不可能同時滿足一致性(C:Consistency)、可用性(A:Availability)和分區(qū)容錯性(P:Partition tolerance)這三個基本需求,最多只能同時滿足其中的兩項。

  一致性  在分布式環(huán)境中,一致性是指數(shù)據(jù)在多個副本之間是否能夠保持一致的特性。在一致性的需求下,當(dāng)一個系統(tǒng)在數(shù)據(jù)一致的狀態(tài)下執(zhí)行更新操作后,應(yīng)該保證系統(tǒng)的數(shù)據(jù)仍然處于一致的狀態(tài)。

  對于一個將數(shù)據(jù)副本分布在不同分布式節(jié)點上的系統(tǒng)來說,如果對第一個節(jié)點的數(shù)據(jù)進行了更新操作并且更新成功后,卻沒有使得第二個節(jié)點上的數(shù)據(jù)得到相應(yīng)的更新,于是在對第二個節(jié)點的數(shù)據(jù)進行讀取操作時,獲取的依然是老數(shù)據(jù)(或稱為臟數(shù)據(jù)),這就是典型的分布式數(shù)據(jù)不一致情況。在分布式系統(tǒng)中,如果能夠做到針對一個數(shù)據(jù)項的更新操作執(zhí)行成功后,所有的用戶都可以讀取到其最新的值,那么這樣的系統(tǒng)就被認為具有強一致性(或嚴格的一致性)。

  可用性

  可用性是指系統(tǒng)提供的服務(wù)必須一直處于可用的狀態(tài),對于用戶的每一個操作請求總是能夠在有限的時間內(nèi)返回結(jié)果。這里我們重點看下“有限的時間內(nèi)”和“返回結(jié)果”。

  “有限的時間內(nèi)”是指,對于用戶的一個操作請求,系統(tǒng)必須能夠在指定的時間(即響應(yīng)時間)內(nèi)返回對應(yīng)的處理結(jié)果,如果超過了這個時間范圍,那么系統(tǒng)就被認為是不可用的。另外,“有限的時間內(nèi)”是一個在系統(tǒng)設(shè)計之初就設(shè)定好的系統(tǒng)運行指標(biāo),通常不同的系統(tǒng)之間會有很大的不同。比如說,對于一個在線搜索引擎來說,通常在0.5秒內(nèi)需要給出用戶搜索關(guān)鍵詞對應(yīng)的檢索結(jié)果。以Google為例,搜索“分布式”這一關(guān)鍵詞,Google能夠在0.3秒左右的時間,返回大約上千萬條檢索結(jié)果。而對于一個面向HIVE的海量數(shù)據(jù)查詢平臺來說,正常的一次數(shù)據(jù)檢索時間可能在20秒到30秒之間,而如果是一個時間跨度較大的數(shù)據(jù)內(nèi)容查詢,“有限的時間”有時候甚至?xí)L達幾分鐘。

  從上面的例子中,我們可以看出,用戶對于一個系統(tǒng)的請求響應(yīng)時間的期望值不盡相同。但是,無論系統(tǒng)之間的差異有多大,唯一相同的一點就是對于用戶請求,系統(tǒng)必須存在一個合理的響應(yīng)時間,否則用戶便會對系統(tǒng)感到失望。

  “返回結(jié)果”是可用性的另一個非常重要的指標(biāo),它要求系統(tǒng)在完成對用戶請求的處理后,返回一個正常的響應(yīng)結(jié)果。正常的響應(yīng)結(jié)果通常能夠明確地反映出對請求的處理結(jié)果,即成功或失敗,而不是一個讓用戶感到困惑的返回結(jié)果。

  讓我們再來看看上面提到的在線搜索引擎的例子,如果用戶輸入指定的搜索關(guān)鍵詞后,返回的結(jié)果是一個系統(tǒng)錯誤,通常類似于“OutOfMemoryError”或“System Has Crashed”等提示語,那么我們認為此時系統(tǒng)是不可用的。

  分區(qū)容錯性

  分區(qū)容錯性約束了一個分布式系統(tǒng)需要具有如下特性:分布式系統(tǒng)在遇到任何網(wǎng)絡(luò)分區(qū)故障的時候,仍然需要能夠保證對外提供滿足一致性和可用性的服務(wù),除非是整個網(wǎng)絡(luò)環(huán)境都發(fā)生了故障。

  網(wǎng)絡(luò)分區(qū)是指在分布式系統(tǒng)中,不同的節(jié)點分布在不同的子網(wǎng)絡(luò)(機房或異地網(wǎng)絡(luò)等)中,由于一些特殊的原因?qū)е逻@些子網(wǎng)絡(luò)之間出現(xiàn)網(wǎng)絡(luò)不連通的狀況,但各個子網(wǎng)絡(luò)的內(nèi)部網(wǎng)絡(luò)是正常的,從而導(dǎo)致整個系統(tǒng)的網(wǎng)絡(luò)環(huán)境被切分成了若干個孤立的區(qū)域。需要注意的是,組成一個分布式系統(tǒng)的每個節(jié)點的加入與退出都可以看作是一個特殊的網(wǎng)絡(luò)分區(qū)。

  以上就是對CAP定理中一致性、可用性和分區(qū)容錯性的講解,通常使用圖1-2所示的示意圖來表示CAP定理。

  既然在上文中我們提到,一個分布式系統(tǒng)無法同時滿足上述三個需求,而只能滿足其中的兩項,因此在進行對CAP定理的應(yīng)用時,我們就需要拋棄其中的一項,表1-2所示是拋棄CAP定理中任意一項特性的場景說明。

  圖1-2.CAP定理示意圖

  表1-2.CAP定理應(yīng)用

  放棄CAP定理 說明

  放棄P 如果希望能夠避免系統(tǒng)出現(xiàn)分區(qū)容錯性問題,一種較為簡單的做法是將所有的數(shù)據(jù)(或者僅僅是那些與事務(wù)相關(guān)的數(shù)據(jù))都放在一個分布式節(jié)點上。這樣的做法雖然無法100%地保證系統(tǒng)不會出錯,但至少不會碰到由于網(wǎng)絡(luò)分區(qū)帶來的負面影響。但同時需要注意的是,放棄P的同時也就意味著放棄了系統(tǒng)的可擴展性

  放棄A 相對于放棄“分區(qū)容錯性”來說,放棄可用性則正好相反,其做法是一旦系統(tǒng)遇到網(wǎng)絡(luò)分區(qū)或其他故障時,那么受到影響的服務(wù)需要等待一定的時間,因此在等待期間系統(tǒng)無法對外提供正常的服務(wù),即不可用

  放棄C 這里所說的放棄一致性,并不是完全不需要數(shù)據(jù)一致性,如果真是這樣的話,那么系統(tǒng)的數(shù)據(jù)都是沒有意義的,整個系統(tǒng)也是沒有價值的。

  事實上,放棄一致性指的是放棄數(shù)據(jù)的強一致性,而保留數(shù)據(jù)的最終一致性。這樣的系統(tǒng)無法保證數(shù)據(jù)保持實時的一致性,但是能夠承諾的是,數(shù)據(jù)最終會達到一個一致的狀態(tài)。這就引入了一個時間窗口的概念,具體多久能夠達到數(shù)據(jù)一致取決于系統(tǒng)的設(shè)計,主要包括數(shù)據(jù)副本在不同節(jié)點之間的復(fù)制時間長短

  從CAP定理中我們可以看出,一個分布式系統(tǒng)不可能同時滿足一致性、可用性和分區(qū)容錯性這三個需求。另一方面,需要明確的一點是,對于一個分布式系統(tǒng)而言,分區(qū)容錯性可以說是一個最基本的要求。為什么這樣說,其實很簡單,因為既然是一個分布式系統(tǒng),那么分布式系統(tǒng)中的組件必然需要被部署到不同的節(jié)點,否則也就無所謂分布式系統(tǒng)了,因此必然出現(xiàn)子網(wǎng)絡(luò)。而對于分布式系統(tǒng)而言,網(wǎng)絡(luò)問題又是一個必定會出現(xiàn)的異常情況,因此分區(qū)容錯性也就成為了一個分布式系統(tǒng)必然需要面對和解決的問題。因此系統(tǒng)架構(gòu)設(shè)計師往往需要把精力花在如何根據(jù)業(yè)務(wù)特點在C(一致性)和A(可用性)之間尋求平衡。

  BASE理論

  BASE是Basically Available(基本可用)、Soft state(軟狀態(tài))和Eventually consistent(最終一致性)三個短語的簡寫,是由來自eBay的架構(gòu)師Dan Pritchett在其文章BASE: An Acid Alternative注 中第一次明確提出的。BASE是對CAP中一致性和可用性權(quán)衡的結(jié)果,其來源于對大規(guī)?;ヂ?lián)網(wǎng)系統(tǒng)分布式實踐的總結(jié),是基于CAP定理逐步演化而來的,其核心思想是即使無法做到強一致性(Strong consistency),但每個應(yīng)用都可以根據(jù)自身的業(yè)務(wù)特點,采用適當(dāng)?shù)姆绞絹硎瓜到y(tǒng)達到最終一致性(Eventual consistency)。接下來我們著重對BASE中的三要素進行詳細講解。

  基本可用

  基本可用是指分布式系統(tǒng)在出現(xiàn)不可預(yù)知故障的時候,允許損失部分可用性——但請注意,這絕不等價于系統(tǒng)不可用。以下兩個就是“基本可用”的典型例子。

  響應(yīng)時間上的損失:正常情況下,一個在線搜索引擎需要在0.5秒之內(nèi)返回給用戶相應(yīng)的查詢結(jié)果,但由于出現(xiàn)故障(比如系統(tǒng)部分機房發(fā)生斷電或斷網(wǎng)故障),查詢結(jié)果的響應(yīng)時間增加到了1~2秒。

  功能上的損失:正常情況下,在一個電子商務(wù)網(wǎng)站上進行購物,消費者幾乎能夠順利地完成每一筆訂單,但是在一些節(jié)日大促購物高峰的時候,由于消費者的購物行為激增,為了保護購物系統(tǒng)的穩(wěn)定性,部分消費者可能會被引導(dǎo)到一個降級頁面。

  弱狀態(tài)

  弱狀態(tài)也稱為軟狀態(tài),和硬狀態(tài)相對,是指允許系統(tǒng)中的數(shù)據(jù)存在中間狀態(tài),并認為該中間狀態(tài)的存在不會影響系統(tǒng)的整體可用性,即允許系統(tǒng)在不同節(jié)點的數(shù)據(jù)副本之間進行數(shù)據(jù)同步的過程存在延時。

  最終一致性

  最終一致性強調(diào)的是系統(tǒng)中所有的數(shù)據(jù)副本,在經(jīng)過一段時間的同步后,最終能夠達到一個一致的狀態(tài)。因此,最終一致性的本質(zhì)是需要系統(tǒng)保證最終數(shù)據(jù)能夠達到一致,而不需要實時保證系統(tǒng)數(shù)據(jù)的強一致性。

  亞馬遜首席技術(shù)官Werner Vogels在于2008年發(fā)表的一篇經(jīng)典文章Eventually Consistent-

  Revisited中,對最終一致性進行了非常詳細的介紹。他認為最終一致性是一種特殊的弱一致性:系統(tǒng)能夠保證在沒有其他新的更新操作的情況下,數(shù)據(jù)最終一定能夠達到一致的狀態(tài),因此所有客戶端對系統(tǒng)的數(shù)據(jù)訪問都能夠獲取到最新的值。同時,在沒有發(fā)生故障的前提下,數(shù)據(jù)達到一致狀態(tài)的時間延遲,取決于網(wǎng)絡(luò)延遲、系統(tǒng)負載和數(shù)據(jù)復(fù)制方案設(shè)計等因素。

  在實際工程實踐中,最終一致性存在以下五類主要變種。

  因果一致性(Causal consistency)

  因果一致性是指,如果進程A在更新完某個數(shù)據(jù)項后通知了進程B,那么進程B之后對該數(shù)據(jù)項的訪問都應(yīng)該能夠獲取到進程A更新后的最新值,并且如果進程B要對該數(shù)據(jù)項進行更新操作的話,務(wù)必基于進程A更新后的最新值,即不能發(fā)生丟失更新情況。與此同時,與進程A無因果關(guān)系的進程C的數(shù)據(jù)訪問則沒有這樣的限制。

  讀己之所寫(Read your writes)

  讀己之所寫是指,進程A更新一個數(shù)據(jù)項之后,它自己總是能夠訪問到更新過的最新值,而不會看到舊值。也就是說,對于單個數(shù)據(jù)獲取者來說,其讀取到的數(shù)據(jù),一定不會比自己上次寫入的值舊。因此,讀己之所寫也可以看作是一種特殊的因果一致性。  會話一致性(Session consistency)

  會話一致性將對系統(tǒng)數(shù)據(jù)的訪問過程框定在了一個會話當(dāng)中:系統(tǒng)能保證在同一個有效的會話中實現(xiàn)“讀己之所寫”的一致性,也就是說,執(zhí)行更能操作之后,客戶端能夠在同一個會話中始終讀取到該數(shù)據(jù)項的最新值。

  單調(diào)讀一致性(Monotonic read consistency)

  單調(diào)讀一致性是指如果一個進程從系統(tǒng)中讀取出一個數(shù)據(jù)項的某個值后,那么系統(tǒng)對于該進程后續(xù)的任何數(shù)據(jù)訪問都不應(yīng)該返回更舊的值。

  單調(diào)寫一致性(Monotonic write consistency)

  單調(diào)寫一致性是指,一個系統(tǒng)需要能夠保證來自同一個進程的寫操作被順序地執(zhí)行。

  以上就是最終一致性的五類常見的變種,在實際系統(tǒng)實踐中,可以將其中的若干個變種互相結(jié)合起來,以構(gòu)建一個具有最終一致性特性的分布式系統(tǒng)。事實上,最終一致性并不是只有那些大型分布式系統(tǒng)才涉及的特性,許多現(xiàn)代的關(guān)系型數(shù)據(jù)庫都采用了最終一致性模型。在現(xiàn)代關(guān)系型數(shù)據(jù)庫中,大多都會采用同步和異步方式來實現(xiàn)主備數(shù)據(jù)復(fù)制技術(shù)。在同步方式中,數(shù)據(jù)的復(fù)制過程通常是更新事務(wù)的一部分,因此在事務(wù)完成后,主備數(shù)據(jù)庫的數(shù)據(jù)就會達到一致。而在異步方式中,備庫的更新往往會存在延時,這取決于事務(wù)日志在主備數(shù)據(jù)庫之間傳輸?shù)臅r間長短,如果傳輸時間過長或者甚至在日志傳輸過程中出現(xiàn)異常導(dǎo)致無法及時將事務(wù)應(yīng)用到備庫上,那么很顯然,從備庫中讀取的數(shù)據(jù)將是舊的,因此就出現(xiàn)了數(shù)據(jù)不一致的情況。當(dāng)然,無論是采用多次重試還是人為數(shù)據(jù)訂正,關(guān)系型數(shù)據(jù)庫還是能夠保證最終數(shù)據(jù)達到一致——這就是系統(tǒng)提供最終一致性保證的經(jīng)典案例。

  總的來說,BASE理論面向的是大型高可用可擴展的分布式系統(tǒng),和傳統(tǒng)事務(wù)的ACID特性是相反的,它完全不同于ACID的強一致性模型,而是提出通過犧牲強一致性來獲得可用性,并允許數(shù)據(jù)在一段時間內(nèi)是不一致的,但最終達到一致狀態(tài)。但同時,在實際的分布式場景中,不同業(yè)務(wù)單元和組件對數(shù)據(jù)一致性的要求是不同的,因此在具體的分布式系統(tǒng)架構(gòu)設(shè)計過程中,ACID特性與BASE理論往往又會結(jié)合在一起使用。

  小結(jié)

  計算機系統(tǒng)從集中式向分布式的變革伴隨著包括分布式網(wǎng)絡(luò)、分布式事務(wù)和分布式數(shù)據(jù)一致性等在內(nèi)的一系列問題與挑戰(zhàn),同時也催生了一大批諸如ACID、CAP和BASE等經(jīng)典理論的快速發(fā)展。

  本章由計算機系統(tǒng)從集中式向分布式發(fā)展的過程展開,圍繞在分布式架構(gòu)發(fā)展過程中碰到的一系列問題,結(jié)合ACID、CAP和BASE等分布式事務(wù)與一致性方面的經(jīng)典理論,向讀者介紹了分布式架構(gòu)。

  ……


前言/序言

問題的提出

  在計算機科學(xué)領(lǐng)域,分布式一致性問題是一個相當(dāng)重要,且被廣泛探索與論證的問題,通常存在于諸如分布式文件系統(tǒng)、緩存系統(tǒng)和數(shù)據(jù)庫等大型分布式存儲系統(tǒng)中。

  什么是分布式一致性?分布式一致性分為哪些類型?分布式系統(tǒng)達到一致性后將會是一個什么樣的狀態(tài)?如果失去了一致性約束,分布式系統(tǒng)是否還可以依賴?如果一味地追求一致性,對系統(tǒng)的整體架構(gòu)和性能又有多大影響?這一系列的問題,似乎都沒有一個嚴格意義上準確的定義和答案。

終端用戶

 IT技術(shù)的發(fā)展,讓我們受益無窮,從日常生活的超市收銀,到高端精細的火箭發(fā)射,現(xiàn)代社會中幾乎所有行業(yè),都離不開計算機技術(shù)的支持。

盡管計算機工程師們創(chuàng)造出了很多高科技的計算機產(chǎn)品來解決我們?nèi)粘E龅降膯栴},但用戶只會傾向于選擇一些易用、好用的產(chǎn)品,那些難以使用的計算機產(chǎn)品最終都會被淘汰——這種易用性,其實就是用戶體驗的一部分。

計算機產(chǎn)品的用戶體驗,可以分為便捷性、安全性和穩(wěn)定性等方面。在本書中,我們主要討論的是用戶在使用計算機產(chǎn)品過程中遇到的那些和一致性有關(guān)的問題。在此之前,我們首先來看一下計算機產(chǎn)品的終端用戶是誰,他們的需求又是什么。

火車站售票

假如說我們的終端用戶是一位經(jīng)常做火車的旅行家,通常他是去車站的售票處購買車票,然后拿著車票去檢票口,再坐上火車,開始一段美好的旅行——一切似乎都是那么和諧。想象一下,如果他選擇的目的地是杭州,而某一趟開往杭州的火車只剩下最后一張車票了,可能在同一時刻,不同售票窗口的另一位乘客也購買了同一張車票。假如說售票系統(tǒng)沒有進行一致性保障,兩人都購票成功了。而在檢票口檢票的時候,其中一位乘客會被告知他的車票無效——當(dāng)然,現(xiàn)代的中國鐵路售票系統(tǒng)已經(jīng)很少出現(xiàn)這樣的問題了,但在這個例子中,我們可以看出,終端用戶對于我們的系統(tǒng)的需求非常簡單:

“請售票給我,如果沒有余票了,請在售票的時候就告訴我票是無效票的?!?/p>

 這就對購票系統(tǒng)提出了嚴格的一致性要求——系統(tǒng)的數(shù)據(jù)(在本例中指的就是那趟開往杭州的火車的余票數(shù)),無論在哪個售票窗口,每時每刻都必須是準確無誤的!

銀行轉(zhuǎn)賬

假如說我們的終端用戶是一名剛畢業(yè)的大學(xué)生,通常在拿到第一個月工資之后,都會選擇向家里匯款。當(dāng)他來到銀行柜臺,完成轉(zhuǎn)賬操作后,銀行的柜臺服務(wù)員會友善地提醒他:“您的轉(zhuǎn)賬將在N個工作日后到賬!”此時這名畢業(yè)生有一些沮喪,會對那名柜臺服務(wù)員叮囑:“好吧,多久沒關(guān)系,錢不要少就行了!”——這也成為了幾乎所有的用戶對于現(xiàn)代銀行系統(tǒng)最基本的需求。

網(wǎng)上購物

假如說我們的終端用戶是一名網(wǎng)上購物狂,當(dāng)他看到一件庫存量為5的心儀商品,會迅速地確認購買,寫下收貨地址,然后下單——然而,在下單的那個瞬間,系統(tǒng)可能會告知該用戶:“庫存量不足!”此時,絕大部分的消費者往往都會抱怨自己動作太慢,使得心愛的商品被其他人搶走了!

但其實有過網(wǎng)購系統(tǒng)開發(fā)經(jīng)驗的工程師一定明白,在商品詳情頁面上顯示的那個庫存量,通常不是該商品的真實庫存量,只有在真正下單購買的時候,系統(tǒng)才會檢查該商品的真實庫存量。但是,誰在意呢?

在上面三個例子中,相信讀者一定已經(jīng)看出來了,我們的終端用戶在使用不同的計算機產(chǎn)品時對于數(shù)據(jù)一致性的需求是不一樣的:

有些系統(tǒng),既要快速地響應(yīng)用戶,同時還要保證系統(tǒng)的數(shù)據(jù)對于任意客戶端都是真實可靠的,就像火車站的售票系統(tǒng)。

還有些系統(tǒng),需要為用戶保證絕對可靠的數(shù)據(jù)安全,雖然在數(shù)據(jù)一致性上存在延時,但最終務(wù)必保證嚴格的一致,就像銀行的轉(zhuǎn)賬系統(tǒng)。

另外的一些系統(tǒng),雖然向用戶展示了一些可以說是“錯誤”的數(shù)據(jù),但是在整個系統(tǒng)使用過程中,一定會在某一個流程上對系統(tǒng)數(shù)據(jù)進行準確無誤的檢查,從而避免用戶發(fā)生不必要的損失,就像網(wǎng)購系統(tǒng)。

更新的并發(fā)性

在計算機發(fā)展的早期階段,受到底層硬件技術(shù)的制約,同時也是由于人們對于計算機系統(tǒng)的實際使用需求比較簡單,因此很多上層的應(yīng)用程序架構(gòu)都是單線程模型的。以C語言為例,其誕生于上世紀70年代,當(dāng)時幾乎所有使用C語言開發(fā)的應(yīng)用程序都是單線程的。從現(xiàn)在來看,單線程應(yīng)用程序雖然在運行效率上無法和后來的多線程應(yīng)用程序相比,但是在編程模型上相對簡單,因此能夠避免多線程程序中出現(xiàn)的不少并發(fā)問題。

隨著計算機底層硬件技術(shù)和現(xiàn)代操作系統(tǒng)的不斷發(fā)展,多線程技術(shù)開始被越來越多地引入到計算機編程模型之中,并對現(xiàn)代計算機應(yīng)用程序的整體架構(gòu)起到了至關(guān)重要的作用。

多線程的引入,為應(yīng)用程序帶來性能上的卓越提升,同時也帶來了一個最大的副作用,那就是并發(fā)?!渡钊肜斫庥嬎銠C系統(tǒng)》一書對并發(fā)進行了如下定義:如果邏輯控制流在時間上重疊,那么它們就是并發(fā)的。這里提到的邏輯控制流,通俗地講,就是一次程序操作,比如讀取或更新內(nèi)存中變量的值。

在本書后面的討論中,我們提到的“并發(fā)”都特指更新操作的并發(fā),即有多個線程同時更新內(nèi)存中變量的值——我們將這一現(xiàn)象稱為更新的并發(fā)性。

分布式一致性問題

在分布式系統(tǒng)中另一個需要解決的重要問題就是數(shù)據(jù)的復(fù)制。在我們?nèi)粘5拈_發(fā)經(jīng)驗中,相信很多開發(fā)人員都碰到過這樣的問題:假設(shè)客戶端C1將系統(tǒng)中的一個值K由V1更新為V2,但客戶端C2無法立即讀取到K的最新值,需要在一段時間之后才能讀取到。讀者可能也已經(jīng)猜到了,上面這個例子就是常見的數(shù)據(jù)庫之間復(fù)制的延時問題。

分布式系統(tǒng)對于數(shù)據(jù)的復(fù)制需求一般都來自于以下兩個原因。

為了增加系統(tǒng)的可用性,以防止單點故障引起的系統(tǒng)不可用。

提高系統(tǒng)的整體性能,通過負載均衡技術(shù),能夠讓分布在不同地方的數(shù)據(jù)副本都能夠為用戶提供服務(wù)。

數(shù)據(jù)復(fù)制在可用性和性能方面給分布式系統(tǒng)帶來的巨大好處是不言而喻的,然而數(shù)據(jù)復(fù)制所帶來的一致性挑戰(zhàn),也是每一個系統(tǒng)研發(fā)人員不得不面對的。

所謂的分布式一致性問題,是指在分布式環(huán)境中引入數(shù)據(jù)復(fù)制機制后,不同數(shù)據(jù)節(jié)點間可能出現(xiàn)的,并無法依靠計算機應(yīng)用程序自身解決的數(shù)據(jù)不一致情況。簡單地講,數(shù)據(jù)一致性就是指在對一個副本數(shù)據(jù)進行更新的同時,必須確保也能夠更新其他的副本,否則不同副本之間的數(shù)據(jù)將不再一致。

那怎么來解決這個問題呢?順著上面提到的復(fù)制延時問題,很快就有人想到了一種解決辦法,那就是:

 “既然是由于延時引起的問題,那我可以將寫入的動作阻塞,直到數(shù)據(jù)復(fù)制完成后,才完成寫入動作?!?/p>

沒錯,這似乎能解決問題,而且有一些系統(tǒng)的架構(gòu)也確實直接使用了這個思路。但這個思路在解決一致性問題的同時,又帶來了新的問題:寫入的性能。如果你的應(yīng)用場景有非常多的寫請求,那么使用這個思路之后,后續(xù)的寫請求都將會阻塞在前一個請求的寫操作上,導(dǎo)致系統(tǒng)整理性能急劇下降。

總的來講,我們無法找到一種能夠滿足分布式系統(tǒng)所有系統(tǒng)屬性的分布式一致性解決方案。因此,如何既保證數(shù)據(jù)的一致性,同時又不影響系統(tǒng)運行的性能,是每一個分布式系統(tǒng)都需要重點考慮和權(quán)衡的。于是,一致性級別由此誕生。

強一致性

這種一致性級別是最符合用戶直覺的,它要求系統(tǒng)寫入什么,讀出來的也會是什么,用戶體驗好,但實現(xiàn)起來往往對系統(tǒng)的性能影響比較大。

弱一致性

這種一致性級別約束了系統(tǒng)在寫入成功后,不承諾立即可以讀到寫入的值,也不具體承諾多久之后數(shù)據(jù)能夠達到一致,但會盡可能地保證到某個時間級別(比如秒級別)后,數(shù)據(jù)能夠達到一致狀態(tài)。弱一致性還可以再進行細分:

會話一致性:該一致性級別只保證對于寫入的值,在同一個客戶端會話中可以讀到一致的值,但其他的會話不能保證。

用戶一致性:該一致性級別只保證對于寫入的值,在同一個用戶中可以讀到一致的值,但其他用戶不能保證。

最終一致性

最終一致性是弱一致性的一個特例,系統(tǒng)會保證在一定時間內(nèi),能夠達到一個數(shù)據(jù)一致的狀態(tài)。這里之所以將最終一致性單獨提出來,是因為它是弱一致性中非常重要的一種一致性模型,也是業(yè)界在大型分布式系統(tǒng)的數(shù)據(jù)一致性上比較推崇的模型。

本書將會從分布式一致性的理論出發(fā),向讀者講解幾種典型的分布式一致性協(xié)議是如何解決分布式一致性問題的。之后,本書則會深入介紹分布式一致性問題的工業(yè)解決方案——ZooKeeper,并著重向讀者展示這一分布式協(xié)調(diào)框架的使用方法、內(nèi)部實現(xiàn)以及運維技巧。

致謝

  首先要感謝現(xiàn)在的部門老大蔣江偉先生。第一次接觸蔣江偉是在2011年,當(dāng)時參加了他的一個講座“淘寶前臺系統(tǒng)優(yōu)化實踐——吞吐量優(yōu)化”,對其中關(guān)于“編寫GC友好代碼”的內(nèi)容有不解之處,于是私下請教。他耐心的講解令我至今記憶猶新。兩年前,他全面負責(zé)中間件團隊之后,給予了我更大的幫助和鼓勵,使我得到了極大的進步,真的非常感謝。本書的問世,離不開他的推薦。也正是這一份寫作的責(zé)任感,讓我有決心和毅力來對整個ZooKeeper內(nèi)容進行了一次全面的整理。在這里,衷心祝福蔣江偉先生帶領(lǐng)中間件團隊走向新的高度。

  其次,本書的寫作,離不開各位小伙伴們的支持和幫助,他們是各領(lǐng)域的資深專家,我向他們征集了很多有營養(yǎng)的內(nèi)容。在這里,按照章節(jié)順序,依次表示感謝:許澤彬參與了“問題提出”的寫作;侯前明對Paxos算法的前世今生進行的整理;段培樂對晦澀的Paxos協(xié)議進行了細致的講解;姜宇向我提供了他對于分布式事務(wù)的見解;徐偉辰參與了分布式鎖服務(wù)Chubby相關(guān)的寫作;葉成旭提供了他在上家公司時對Hypertable的學(xué)習(xí)和研究成果;高偉細致地向我展示了Curator這一ZooKeeper客戶端的使用;陳杰提供了他在“自動化的DNS服務(wù)”場景中的經(jīng)驗總結(jié);曹龍參與了Hadoop相關(guān)內(nèi)容的寫作;鄧明鑒則貢獻了他對HBase的深刻見解;作為產(chǎn)品的開源負責(zé)人,莊曉丹和王強提供了對消息中間件Metamorphosis技術(shù)架構(gòu)的講解;李鼎則向我全面展示了RPC服務(wù)框架Dubbo的技術(shù)細節(jié);樓江航向我提供了Canal和Otter這兩個分布式產(chǎn)品中的ZooKeeper應(yīng)用場景;李雨前、柳明和溫朝凱則一起寫了終搜在產(chǎn)品演進過程中對ZooKeeper的使用和改進;封仲淹參與了對其自主產(chǎn)品JStorm的技術(shù)剖析……是你們一遍又一遍地對內(nèi)容進行修改,才使得本書內(nèi)容更為豐滿。

  另外,也要感謝溫文鎏、王林、許澤彬、高偉和段培樂等人對全書的審閱,正是你們提出的寶貴建議,對完善本書提供了非常大的幫助。

  感謝現(xiàn)在的同事陸學(xué)慧先生,從2013年下半年開始,他全面接手對ZooKeeper的開發(fā)和運維,在他身上感受到的專業(yè)和創(chuàng)新精神讓我備受鼓舞。

  另外,感謝我的第一個主管馬震先生,是他的幫助為我指引了方向,讓我有機會進入ZooKeeper的世界,并負責(zé)這個產(chǎn)品在公司的發(fā)展。盡管由于業(yè)務(wù)調(diào)整,馬震先生已經(jīng)轉(zhuǎn)崗到其他部門,但依然由衷祝福他工作順利。

  還要感謝我的同事,阿里巴巴店鋪平臺的侯前明先生。本來該書作者應(yīng)該是我們兩個人,但是由于期間他的家庭又增加了一個小生命,導(dǎo)致其不得不中途退出。從本書的選題到寫作大綱的制定,他都傾注了不少心血,相信如果有他一起創(chuàng)作,本書內(nèi)容會更加豐滿、深刻。這里表達遺憾的同時,也向這位兩個孩子的父親送去祝福,祝愿他生活美滿。

  感謝本書的責(zé)任編輯劉蕓女士,是她反復(fù)審稿和編排,才能讓本書的內(nèi)容趨于完美。

  感謝本書的封面設(shè)計吳海燕女士,她的努力已經(jīng)無需言表,在技術(shù)書上的這一前衛(wèi)、極富視覺沖擊力的封面設(shè)計,深深震撼到了我,也希望讀者朋友們能夠喜歡。

  尤其感謝本書的策劃編輯張春雨先生。作為一個南方人,我很少有機會和那些有著一口北方腔的朋友交談,第一次接到張春雨先生電話的時候,我才真正領(lǐng)略了北京腔,也正是他的邀請,才能讓我有機會進行本書的撰寫,同時在前后將近1年半的漫長寫作過程中,也是他的幫助和鼓勵,才讓我堅持完成并不斷完善本書的內(nèi)容。在這里,也衷心祝愿張春雨先生事業(yè)更上一層樓。

  最后,還有我的父母,在過去的1年時間里,多次放假沒有回家,盡管父母一直鼓勵我專注工作,專注于自己的事業(yè),但我深知他們內(nèi)心對兒子的牽掛,在這里也深深地向他們道一聲:“謝謝”,也謹以此書獻給我最親愛的爸爸媽媽。

  倪超

  2014年12月于杭州淘寶城


點此購買


以上內(nèi)容是否對您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號
微信公眾號

編程獅公眾號