摘要:可以發(fā)現(xiàn),整個(gè)同步過程是依賴于來進(jìn)行的。不考慮導(dǎo)致的問題,正常的應(yīng)用升級也會(huì)導(dǎo)致應(yīng)用中斷運(yùn)行。注意事項(xiàng)為了避免被回滾的更新被發(fā)布出去,選擇只在一個(gè)變更到達(dá)大多數(shù)節(jié)點(diǎn)不可能被回滾時(shí),才會(huì)將這些變更發(fā)布到應(yīng)用。
Change Stream是MongoDB從3.6開始支持的新特性。這個(gè)新特性有哪些奇妙之處,會(huì)給我們帶來什么便利?本次的文章將就這個(gè)主題進(jìn)行初步討論。
Change Stream是什么?顧名思義,Change Stream即變更流,是MongoDB向應(yīng)用發(fā)布數(shù)據(jù)變更的一種方式。即當(dāng)數(shù)據(jù)庫中有任何數(shù)據(jù)發(fā)生變化,應(yīng)用端都可以得到通知。我們可以將其理解為在應(yīng)用中執(zhí)行的觸發(fā)器。至于應(yīng)用想得到什么數(shù)據(jù),以什么形式得到數(shù)據(jù),則可以通過聚合框架加以過濾和轉(zhuǎn)換。這點(diǎn)將在后文中討論。
Change Stream的原理我們先來回顧一下MongoDB復(fù)制集大致是如何工作的:
應(yīng)用通過驅(qū)動(dòng)向數(shù)據(jù)庫發(fā)起寫入請求;
在同一個(gè)事務(wù)中,MongoDB完成oplog和集合的修改;
oplog被其他從節(jié)點(diǎn)拉走;
從節(jié)點(diǎn)應(yīng)用得到的oplog,同樣在一個(gè)事務(wù)中完成對oplog和集合的修改;
至此,復(fù)制集同步完成??梢园l(fā)現(xiàn),整個(gè)同步過程是依賴于oplog來進(jìn)行的。也就是說oplog實(shí)際上已經(jīng)包含了我們需要的所有變更數(shù)據(jù)。如果觀測oplog的變化,是否就能夠得到所有變更的數(shù)據(jù)了呢?對,change stream正是基于這個(gè)原理實(shí)現(xiàn)的。但事情并沒有這么簡單!我們來看一下問題有可能出在什么地方。
如何從斷點(diǎn)恢復(fù)現(xiàn)實(shí)世界中,沒有哪個(gè)應(yīng)用是可以不間斷運(yùn)行的。不考慮bug導(dǎo)致的問題,正常的應(yīng)用升級也會(huì)導(dǎo)致應(yīng)用中斷運(yùn)行。那么在應(yīng)用恢復(fù)的時(shí)候,從哪里開始繼續(xù)獲取變更呢?oplog當(dāng)然是可以幫我們做到這點(diǎn)的,但你必須對MongoDB足夠了解,才知道有oplogReplay這樣的參數(shù),以及其他一些問題。
如何有效地處理訂閱假設(shè)在一個(gè)應(yīng)用中需要訂閱10個(gè)不同集合的變更情況,是否需要開10個(gè)tailable cursor去獲取oplog的變更呢?如果是100個(gè)集合呢?出于效率考慮顯然不應(yīng)該這么做。那么整個(gè)過程就會(huì)變成一個(gè)生產(chǎn)者-消費(fèi)者模式,由一個(gè)線程負(fù)責(zé)從oplog獲取變更,由訂閱的線程負(fù)責(zé)消費(fèi)這些變更。雖然實(shí)現(xiàn)也不是那么復(fù)雜,并且多半可以找到開源實(shí)現(xiàn),但是涉及多線程就已經(jīng)足夠讓初學(xué)者頭疼一陣的了。
公平地說,上面這些還不算嚴(yán)重的問題,下面這些問題可能會(huì)更讓人頭疼。
想要tail oplog,必須對local.oplog.rs有讀權(quán)限。實(shí)際上這相當(dāng)于對整個(gè)數(shù)據(jù)庫都有了讀權(quán)限,因?yàn)樗械淖兏紩?huì)在這里體現(xiàn)出來。DBA可能會(huì)阻止你這么做,因?yàn)檫@實(shí)在不是一個(gè)很安全的做法。
如何數(shù)據(jù)回滾極端情況下,如果應(yīng)用處理不當(dāng),MongoDB中可能發(fā)生數(shù)據(jù)回滾rollback的問題。如果僅僅通過跟蹤oplog,則會(huì)出現(xiàn)已經(jīng)通知出去的變更被回滾的情況。
幸運(yùn)的是上面這些問題現(xiàn)在都不是問題了,因?yàn)閏hange stream幫我們規(guī)避了這些復(fù)雜的細(xì)節(jié)。
使用方法由于各種驅(qū)動(dòng)都會(huì)有不同的語法和API,從shell中嘗試使用change stream可能是最簡便的方法。這并不妨礙你隨后在各種驅(qū)動(dòng)中的使用,因?yàn)閟hell中能實(shí)現(xiàn)的功能在驅(qū)動(dòng)中一定有對應(yīng)的語法。下面就以shell為例看看change stream應(yīng)該如何使用。
打開一個(gè)shell,訂閱你需要關(guān)注的集合
比如:
var cursor = db.bar.watch();
為了便于演示,我們在這個(gè)shell中不斷遍歷這個(gè)游標(biāo)以獲取新數(shù)據(jù):
while(true) { if (cursor.hasNext()) { print(JSON.stringify(cursor.next())); } }
打開另一個(gè)shell,向bar集合中插入一條數(shù)據(jù):
db.bar.insert({y: 1})
此時(shí)第一個(gè)shell中會(huì)立即輸出變更數(shù)據(jù):
{"_id":{"_data":{"$binary":"glzquiIAAAACRmRfaWQAZFzquiK0lDNo+K0DpwBaEARUMrm0ruVACoftuxjt1RtCBA==","$type":"00"}},"operationType":"insert","fullDocument":{"_id":{"$oid":"5ceaba22b4943368f8ad03a7"},"y":1},"ns":{"db":"test","coll":"bar"},"documentKey":{"_id":{"$oid":"5ceaba22b4943368f8ad03a7"}}}
這里的一些字段的簡單介紹。更完整的介紹請查閱文檔change events:
_id: 用于恢復(fù)斷點(diǎn)時(shí)使用。即知道這個(gè)值,應(yīng)用斷開后下次重啟里就可以從這個(gè)斷點(diǎn)之后開始恢復(fù)獲得變更;
operationType: 操作類型,常見的值包括:
insert
update
delete
ns: 正在操作的命名空間
fullDocument: 完整的文檔
從斷點(diǎn)恢復(fù)
var cursor = db.bar.watch([], {resumeAfter: <\_id>})
此時(shí)使用hasNext()/next()即可獲取到隨后的變更。
注意事項(xiàng) {readConcern: "majority"}為了避免被回滾的更新被發(fā)布出去,change stream選擇只在一個(gè)變更到達(dá)大多數(shù)節(jié)點(diǎn)(不可能被回滾)時(shí),才會(huì)將這些變更發(fā)布到應(yīng)用。使用的方式即{readConcern: "majority"}。因此以下這些情況下change stream都是不會(huì)向應(yīng)用通知任何變更的:
禁用了readConcern;
從舊版本升級,但沒有更新featureCompatibilityVersion;
PSA架構(gòu)中S宕機(jī);
斷點(diǎn)可恢復(fù)時(shí)間因?yàn)閏hange stream是依賴于oplog工作的,自然也會(huì)面臨oplog面臨的所有問題。問題之一就是oplog被覆蓋。因此想要保證斷點(diǎn)可以恢復(fù),必須保證應(yīng)用在oplog window的時(shí)間內(nèi)請求斷點(diǎn)。
刪除集合如果在訂閱集合變更過程中集合被刪除,則會(huì)收到一條invalid信息通知,表示集合已不再可用:
{ "_id" : { "_data" : BinData(0,"glzqxCcAAAACFFoQBFQyubSu5UAKh+27GO3VG0IE") }, "operationType" : "invalidate" }參考資料
Tailable cursor: https://docs.mongodb.com/manual/core/tailable-cursors/
生產(chǎn)者-消費(fèi)者模式: https://zh.wikipedia.org/wiki/%E7%94%9F%E4%BA%A7%E8%80%85%E6%B6%88%E8%B4%B9%E8%80%85%E9%97%AE%E9%A2%98
關(guān)于回滾: https://docs.mongodb.com/manual/core/replica-set-rollbacks/
變更事件: https://docs.mongodb.com/manual/reference/change-events/
Change Stream介紹文檔:https://docs.mongodb.com/manual/changeStreams/
作者簡介張耀星,MongoDB亞太區(qū)首席技術(shù)咨詢服務(wù)顧問。在MongoDB的開發(fā)、應(yīng)用和咨詢服務(wù)上有多年實(shí)踐經(jīng)驗(yàn)。作為MongoDB認(rèn)證專家,曾經(jīng)為不同行業(yè)的各類大型客戶提供過培訓(xùn)、性能調(diào)優(yōu)、架構(gòu)設(shè)計(jì)等各類MongoDB相關(guān)技術(shù)服務(wù)。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/19547.html
摘要:第一個(gè)函數(shù)生成一個(gè)新的實(shí)例第二個(gè)函數(shù)接受兩個(gè)參數(shù),第一個(gè)是前面生成的對象,二個(gè)是中包含的元素,函數(shù)體就是把中的元素加入對象中。 感謝同事【天錦】的投稿。投稿請聯(lián)系 [email protected] 上篇文章[Java8初體驗(yàn)(一)lambda表達(dá)式語法]()比較詳細(xì)的介紹了lambda表達(dá)式的方方面面,細(xì)心的讀者會(huì)發(fā)現(xiàn)那篇文章的例子中有很多Stream的例子。這些Stream的例子可...
摘要:原文來源全棧初體驗(yàn)前言據(jù)說現(xiàn)在不會(huì)點(diǎn)后臺的前端都找不到工作了嚇得我這幾天看起了和并且做了一個(gè)應(yīng)該算是最簡單的前后端例子,如圖輸入賬戶密碼,提交表單,保存信息到數(shù)據(jù)庫再重定向到頁面獲取數(shù)據(jù)庫中的信息,渲染在瀏覽器上具體代碼主要技術(shù)前端模板后臺 原文來源: 全棧初體驗(yàn) 前言 據(jù)說現(xiàn)在不會(huì)點(diǎn)后臺的前端都找不到工作了嚇得我這幾天看起了Nodejs和MongoDB并且做了一個(gè)應(yīng)該算是最簡單的前后...
摘要:原文來源全棧初體驗(yàn)前言據(jù)說現(xiàn)在不會(huì)點(diǎn)后臺的前端都找不到工作了嚇得我這幾天看起了和并且做了一個(gè)應(yīng)該算是最簡單的前后端例子,如圖輸入賬戶密碼,提交表單,保存信息到數(shù)據(jù)庫再重定向到頁面獲取數(shù)據(jù)庫中的信息,渲染在瀏覽器上具體代碼主要技術(shù)前端模板后臺 原文來源: 全棧初體驗(yàn) 前言 據(jù)說現(xiàn)在不會(huì)點(diǎn)后臺的前端都找不到工作了嚇得我這幾天看起了Nodejs和MongoDB并且做了一個(gè)應(yīng)該算是最簡單的前后...
摘要:原文來源全棧初體驗(yàn)前言據(jù)說現(xiàn)在不會(huì)點(diǎn)后臺的前端都找不到工作了嚇得我這幾天看起了和并且做了一個(gè)應(yīng)該算是最簡單的前后端例子,如圖輸入賬戶密碼,提交表單,保存信息到數(shù)據(jù)庫再重定向到頁面獲取數(shù)據(jù)庫中的信息,渲染在瀏覽器上具體代碼主要技術(shù)前端模板后臺 原文來源: 全棧初體驗(yàn) 前言 據(jù)說現(xiàn)在不會(huì)點(diǎn)后臺的前端都找不到工作了嚇得我這幾天看起了Nodejs和MongoDB并且做了一個(gè)應(yīng)該算是最簡單的前后...
摘要:前言筆者的前端開發(fā)已經(jīng)有些時(shí)日了,對于一直保留著最初的恐懼,倘若一座不可跨越的高山,思前想后終于邁出最后一步,踏入了開拓自己視野的新視界,希望在看這篇文章的你可以一起跟我動(dòng)手嘗試。面向的下一代框架。由團(tuán)隊(duì)打造,特點(diǎn)優(yōu)雅簡潔靈活體積小。 showImg(https://segmentfault.com/img/bVbuorM?w=1514&h=568); 前言 ?????筆者的前端開發(fā)已...
閱讀 1927·2021-11-22 09:34
閱讀 1158·2021-10-09 09:44
閱讀 3051·2021-09-29 09:35
閱讀 3629·2021-09-14 18:01
閱讀 1497·2021-08-16 10:49
閱讀 1097·2019-08-29 14:11
閱讀 861·2019-08-29 12:47
閱讀 3082·2019-08-26 13:47