亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 數據庫 > MongoDB > 正文

MongoDB數據庫兩階段提交實現事務的方法詳解

2020-10-29 18:43:34
字體:
來源:轉載
供稿:網友

本文實例講述了MongoDB數據庫兩階段提交實現事務的方法。分享給大家供大家參考,具體如下:

MongoDB數據庫中操作單個文檔總是原子性的,然而,涉及多個文檔的操作,通常被作為一個“事務”,而不是原子性的。因為文檔可以是相當復雜并且包含多個嵌套文檔,單文檔的原子性對許多實際用例提供了支持。盡管單文檔操作是原子性的,在某些情況下,需要多文檔事務。在這些情況下,使用兩階段提交,提供這些類型的多文檔更新支持。因為文檔可以表示為Pending數據和狀態,可以使用一個兩階段提交確保數據是一致的,在一個錯誤的情況下,事務前的狀態是可恢復的。

事務最常見的例子是以可靠的方式從A賬戶轉賬到B賬戶,在關系型數據庫中,此操作將從A賬戶減掉金額和給B賬戶增加金額的操作封裝在單個原子事務中。在MongoDB中,可以使用兩階段提交達到相同的效果。本文中的所有示例使用mongo shell與數據庫進行交互,并假設有兩個集合:首先,一個名為accounts的集合存儲每個賬戶的文檔數據,另一個名為transactions的集合存儲事務本身。

首先創建兩個名為A和B的賬戶,使用下面的命令:

db.accounts.save({name: "A", balance: 1000, pendingTransactions: []})db.accounts.save({name: "B", balance: 1000, pendingTransactions: []})

使用find()方法驗證這兩個操作已經成功:

db.accounts.find()

mongo會返回兩個類似下面的文檔:

{ "_id" : ObjectId("4d7bc66cb8a04f512696151f"), "name" : "A", "balance" : 1000, "pendingTransactions" : [ ] }{ "_id" : ObjectId("4d7bc67bb8a04f5126961520"), "name" : "B", "balance" : 1000, "pendingTransactions" : [ ] }

事務過程:

設置事務初始狀態initial:

通過插入下面的文檔創建transaction集合,transaction文檔持有源(source)和目標(destination),它們引用自accounts集合文檔的字段名,以及value字段表示改變balance字段數量的數據。最后,state字段反映事務的當前狀態。

復制代碼 代碼如下:
db.transactions.save({source: "A", destination: "B", value: 100, state: "initial"})

驗證這個操作已經成功,使用find()

db.transactions.find()

這個操作會返回一個類似下面的文檔:

復制代碼 代碼如下:
{ "_id" : ObjectId("4d7bc7a8b8a04f5126961522"), "source" : "A", "destination" : "B", "value" : 100, "state" : "initial" }

切換事務到Pending狀態:

在修改accounts集合記錄之前,將事務狀態從initial設置為pending。使用findOne()方法將transaction文檔賦值給shell會話中的局部變量t:

t = db.transactions.findOne({state: "initial"})

變量t創建后,shell將返回它的值,將會看到如下的輸出:

復制代碼 代碼如下:
{ "_id" : ObjectId("4d7bc7a8b8a04f5126961522"), "source" : "A", "destination" : "B", "value" : 100, "state" : "initial" }

使用update()改變state的值為pending:

db.transactions.update({_id: t._id}, {$set: {state: "pending"}})db.transactions.find()

find()操作將返回transaction集合的內容,類似下面:

復制代碼 代碼如下:
{ "_id" : ObjectId("4d7bc7a8b8a04f5126961522"), "source" : "A", "destination" : "B", "value" : 100, "state" : "pending" }

將事務應用到兩個賬戶:

使用update()方法應用事務到兩個賬戶。在update()查詢中,條件pendingTransactions:{$ne:t._id}阻止事務更新賬戶,如果賬戶的pendingTransaction字段包含事務t的_id:

db.accounts.update( { name: t.source, pendingTransactions: { $ne: t._id } }, { $inc: { balance: -t.value }, $push: { pendingTransactions: t._id } })db.accounts.update( { name: t.destination, pendingTransactions: { $ne: t._id } }, { $inc: { balance: t.value }, $push: { pendingTransactions: t._id } })db.accounts.find()

find()操作將返回accounts集合的內容,現在應該類似于下面的內容:

{ "_id" : ObjectId("4d7bc97fb8a04f5126961523"), "balance" : 900, "name" : "A", "pendingTransactions" : [ ObjectId("4d7bc7a8b8a04f5126961522") ] }{ "_id" : ObjectId("4d7bc984b8a04f5126961524"), "balance" : 1100, "name" : "B", "pendingTransactions" : [ ObjectId("4d7bc7a8b8a04f5126961522") ] }

設置事務狀態為committed:

使用下面的update()操作設置事務的狀態為committed:

db.transactions.update({_id: t._id}, {$set: {state: "committed"}})db.transactions.find()

find()操作發回transactions集合的內容,現在應該類似下面的內容:

復制代碼 代碼如下:
{ "_id" : ObjectId("4d7bc7a8b8a04f5126961522"), "destination" : "B", "source" : "A", "state" : "committed", "value" : 100 }

移除pending事務:

使用下面的update()操作從accounts集合中移除pending事務:

db.accounts.update({name: t.source}, {$pull: {pendingTransactions: t._id}})db.accounts.update({name: t.destination}, {$pull: {pendingTransactions: t._id}})db.accounts.find()

find()操作返回accounts集合內容,現在應該類似下面內容:

{ "_id" : ObjectId("4d7bc97fb8a04f5126961523"), "balance" : 900, "name" : "A", "pendingTransactions" : [ ] }{ "_id" : ObjectId("4d7bc984b8a04f5126961524"), "balance" : 1100, "name" : "B", "pendingTransactions" : [ ] }

設置事務狀態為done:

通過設置transaction文檔的state為done完成事務:

db.transactions.update({_id: t._id}, {$set: {state: "done"}})db.transactions.find()

find()操作返回transaction集合的內容,此時應該類似下面:

復制代碼 代碼如下:
{ "_id" : ObjectId("4d7bc7a8b8a04f5126961522"), "destination" : "B", "source" : "A", "state" : "done", "value" : 100 }

從失敗場景中恢復:

最重要的部分不是上面的典型例子,而是從各種失敗場景中恢復未完成的事務的可能性。這部分將概述可能的失敗,并提供方法從這些事件中恢復事務。這里有兩種類型的失?。?/p>

1、所有發生在第一步(即設置事務的初始狀態initial)之后,但在第三步(即應用事務到兩個賬戶)之前的失敗。為了還原事務,應用應該獲取一個pending狀態的transaction列表并且從第二步(即切換事務到pending狀態)中恢復。

2、所有發生在第三步之后(即應用事務到兩個賬戶)但在第五步(即設置事務狀態為done)之前的失敗。為了還原事務,應用需要獲取一個committed狀態的事務列表,并且從第四步(即移除pending事務)恢復。

因此應用程序總是能夠恢復事務,最終達到一個一致的狀態。應用程序開始捕獲到每個未完成的事務時運行下面的恢復操作。你可能還希望定期運行恢復操作,以確保數據處于一致狀態。達成一致狀態所需要的時間取決于應用程序需要多長時間恢復每個事務。

回滾:

在某些情況下可能需要“回滾”或“撤消”事務,當應用程序需要“取消”該事務時,或者是因為它永遠需要恢復當其中一個帳戶不存在的情況下,或停止現有的事務。這里有兩種可能的回滾操作:

1、應用事務(即第三步)之后,你已經完全提交事務,你不應該回滾事務。相反,創建一個新的事務,切換源(源)和目標(destination)的值。

2、創建事務(即第一步)之后,在應用事務(即第三步)之前,使用下面的處理過程:

設置事務狀態為canceling:

首先設置事務狀態為canceling,使用下面的update()操作:

db.transactions.update({_id: t._id}, {$set: {state: "canceling"}})

撤銷事務:

使用下面的操作順序從兩個賬戶中撤銷事務:

db.accounts.update({name: t.source, pendingTransactions: t._id}, {$inc: {balance: t.value}, $pull: {pendingTransactions: t._id}})db.accounts.update({name: t.destination, pendingTransactions: t._id}, {$inc: {balance: -t.value}, $pull: {pendingTransactions: t._id}})db.accounts.find()

find()操作返回acounts集合的內容,應該類似下面:

{ "_id" : ObjectId("4d7bc97fb8a04f5126961523"), "balance" : 1000, "name" : "A", "pendingTransactions" : [ ] }{ "_id" : ObjectId("4d7bc984b8a04f5126961524"), "balance" : 1000, "name" : "B", "pendingTransactions" : [ ] }

設置事務狀態為canceled:

最后,使用下面的update()狀態將事務狀態設置為canceled:

db.transactions.update({_id: t._id}, {$set: {state: "canceled"}})

參考資料:http://docs.mongodb.org/manual/tutorial/perform-two-phase-commits/

希望本文所述對大家MongoDB數據庫程序設計有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品久久久久久av| 亚洲男人天堂手机在线| 亚洲第一精品久久忘忧草社区| 日韩一区二区三区xxxx| 韩剧1988在线观看免费完整版| 中文字幕亚洲无线码a| 久久久久这里只有精品| 亚洲国产精品资源| 国产亚洲精品久久久| 亚洲欧美日韩精品久久亚洲区| 精品中文视频在线| 精品国产电影一区| 97视频在线看| 国产午夜精品一区理论片飘花| 国产视频在线一区二区| 欧美激情亚洲精品| 国产免费成人av| 亚洲国产精品成人精品| 亚洲精品成人免费| 欧美乱大交做爰xxxⅹ性3| 尤物yw午夜国产精品视频| 亚洲电影免费观看高清完整版在线观看| 66m—66摸成人免费视频| 欧美极度另类性三渗透| 久久精品在线播放| 欧美精品久久久久久久免费观看| 欧美日韩福利电影| 成人a免费视频| 精品国产一区久久久| 亚洲国产一区二区三区在线观看| 久久人人97超碰精品888| 91九色精品视频| 91久久久精品| 久久免费视频在线观看| 中文字幕亚洲一区二区三区| 青青久久av北条麻妃海外网| 国产精品免费小视频| 日韩专区中文字幕| 国产日产久久高清欧美一区| 欧美性色xo影院| 国内精品小视频在线观看| 久久久久久久电影一区| 亚洲欧美日韩高清| 最近更新的2019中文字幕| 精品中文字幕久久久久久| 欧美精品亚州精品| 色av吧综合网| 亚洲精品www久久久久久广东| 最近中文字幕mv在线一区二区三区四区| 欧美高清视频在线| www.欧美三级电影.com| 午夜欧美大片免费观看| 欧美极品少妇xxxxx| 欧美黄色片在线观看| 日韩欧美在线免费| 欧美尤物巨大精品爽| 伦理中文字幕亚洲| 97在线免费观看| 97婷婷大伊香蕉精品视频| 亚洲综合自拍一区| 亚洲第一av网| 亚洲男人的天堂在线播放| 2021国产精品视频| 亚洲国产91精品在线观看| 日韩精品极品在线观看播放免费视频| 在线播放精品一区二区三区| 亚洲男人天堂久| 91香蕉嫩草神马影院在线观看| 久久久视频在线| 日韩风俗一区 二区| 55夜色66夜色国产精品视频| 成人做爽爽免费视频| 日韩av最新在线| 午夜精品一区二区三区在线| 一区二区三区视频免费在线观看| 九九热精品视频在线播放| 亚洲国产高潮在线观看| 亚洲老板91色精品久久| 在线精品国产欧美| 国产精品福利无圣光在线一区| 亚洲精品美女在线| 北条麻妃一区二区三区中文字幕| 国产成人在线一区| 97精品视频在线| 欧美一级淫片丝袜脚交| 亚洲日本中文字幕| 日韩精品视频在线观看网址| 国产91免费观看| 91欧美激情另类亚洲| 在线电影欧美日韩一区二区私密| 国产91精品久久久久久| 在线视频欧美性高潮| 久久久久久久久久久av| 91在线视频免费| www.日本久久久久com.| 日韩精品在线免费播放| 一区二区三区国产视频| 欧美激情精品久久久久久免费印度| 91精品国产综合久久久久久蜜臀| 亚洲国产日韩欧美综合久久| 亚洲国产精品成人精品| 久久成年人视频| 亚洲精品按摩视频| 亚洲欧美在线免费观看| 成人疯狂猛交xxx| 国产视频999| 亚洲色图15p| 国产中文字幕亚洲| 国产999视频| 91精品国产综合久久香蕉的用户体验| 91免费看国产| 日韩精品视频观看| 欧美日韩亚洲成人| 色偷偷888欧美精品久久久| 久久成人免费视频| 国产精品三级美女白浆呻吟| 国内精品久久久久影院 日本资源| 欧美性xxxx极品高清hd直播| 在线免费看av不卡| 欧美日韩亚洲精品一区二区三区| 在线观看日韩av| 亚洲精品98久久久久久中文字幕| 日韩hd视频在线观看| 欧美国产精品日韩| 在线看欧美日韩| 国产精品一香蕉国产线看观看| 在线免费观看羞羞视频一区二区| 456亚洲影院| 亚洲人成电影在线观看天堂色| 日韩视频精品在线| 最近中文字幕mv在线一区二区三区四区| 亚洲视频在线视频| 日韩大片免费观看视频播放| 国产精品尤物福利片在线观看| 国产成人精品日本亚洲| 91人成网站www| 亚洲国产精品成人va在线观看| 精品中文字幕在线2019| 精品无人区乱码1区2区3区在线| 国产精品偷伦视频免费观看国产| 欧美成人中文字幕在线| 青青a在线精品免费观看| 亚洲一区二区三区777| 成人国产精品日本在线| 中文字幕久久久| 久久视频在线直播| 精品国偷自产在线视频99| 欧美专区在线观看| 国产最新精品视频| 亚洲图片欧美午夜| 中文字幕精品www乱入免费视频| 中文字幕日韩精品在线观看| 在线看日韩欧美| 国产午夜精品全部视频在线播放| 欧美精品videossex性护士| 日韩中文在线不卡| 欧美激情va永久在线播放| 一区二区在线视频播放| 最新日韩中文字幕| 欧美激情精品久久久久久大尺度| 国产精选久久久久久| 国产精品96久久久久久| 欧美日韩精品在线视频| 精品国产鲁一鲁一区二区张丽|