国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院

首頁(yè) > 數(shù)據(jù)庫(kù) > MongoDB > 正文

MongoDB數(shù)據(jù)庫(kù)兩階段提交實(shí)現(xiàn)事務(wù)的方法詳解

2020-10-29 18:43:34
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

本文實(shí)例講述了MongoDB數(shù)據(jù)庫(kù)兩階段提交實(shí)現(xiàn)事務(wù)的方法。分享給大家供大家參考,具體如下:

MongoDB數(shù)據(jù)庫(kù)中操作單個(gè)文檔總是原子性的,然而,涉及多個(gè)文檔的操作,通常被作為一個(gè)“事務(wù)”,而不是原子性的。因?yàn)槲臋n可以是相當(dāng)復(fù)雜并且包含多個(gè)嵌套文檔,單文檔的原子性對(duì)許多實(shí)際用例提供了支持。盡管單文檔操作是原子性的,在某些情況下,需要多文檔事務(wù)。在這些情況下,使用兩階段提交,提供這些類(lèi)型的多文檔更新支持。因?yàn)槲臋n可以表示為Pending數(shù)據(jù)和狀態(tài),可以使用一個(gè)兩階段提交確保數(shù)據(jù)是一致的,在一個(gè)錯(cuò)誤的情況下,事務(wù)前的狀態(tài)是可恢復(fù)的。

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

首先創(chuàng)建兩個(gè)名為A和B的賬戶,使用下面的命令:

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

使用find()方法驗(yàn)證這兩個(gè)操作已經(jīng)成功:

db.accounts.find()

mongo會(huì)返回兩個(gè)類(lèi)似下面的文檔:

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

事務(wù)過(guò)程:

設(shè)置事務(wù)初始狀態(tài)initial:

通過(guò)插入下面的文檔創(chuàng)建transaction集合,transaction文檔持有源(source)和目標(biāo)(destination),它們引用自accounts集合文檔的字段名,以及value字段表示改變balance字段數(shù)量的數(shù)據(jù)。最后,state字段反映事務(wù)的當(dāng)前狀態(tài)。

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

驗(yàn)證這個(gè)操作已經(jīng)成功,使用find()

db.transactions.find()

這個(gè)操作會(huì)返回一個(gè)類(lèi)似下面的文檔:

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

切換事務(wù)到Pending狀態(tài):

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

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

變量t創(chuàng)建后,shell將返回它的值,將會(huì)看到如下的輸出:

復(fù)制代碼 代碼如下:
{ "_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集合的內(nèi)容,類(lèi)似下面:

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

將事務(wù)應(yīng)用到兩個(gè)賬戶:

使用update()方法應(yīng)用事務(wù)到兩個(gè)賬戶。在update()查詢中,條件pendingTransactions:{$ne:t._id}阻止事務(wù)更新賬戶,如果賬戶的pendingTransaction字段包含事務(wù)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集合的內(nèi)容,現(xiàn)在應(yīng)該類(lèi)似于下面的內(nèi)容:

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

設(shè)置事務(wù)狀態(tài)為committed:

使用下面的update()操作設(shè)置事務(wù)的狀態(tài)為committed:

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

find()操作發(fā)回transactions集合的內(nèi)容,現(xiàn)在應(yīng)該類(lèi)似下面的內(nèi)容:

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

移除pending事務(wù):

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

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集合內(nèi)容,現(xiàn)在應(yīng)該類(lèi)似下面內(nèi)容:

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

設(shè)置事務(wù)狀態(tài)為done:

通過(guò)設(shè)置transaction文檔的state為done完成事務(wù):

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

find()操作返回transaction集合的內(nèi)容,此時(shí)應(yīng)該類(lèi)似下面:

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

從失敗場(chǎng)景中恢復(fù):

最重要的部分不是上面的典型例子,而是從各種失敗場(chǎng)景中恢復(fù)未完成的事務(wù)的可能性。這部分將概述可能的失敗,并提供方法從這些事件中恢復(fù)事務(wù)。這里有兩種類(lèi)型的失敗:

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

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

因此應(yīng)用程序總是能夠恢復(fù)事務(wù),最終達(dá)到一個(gè)一致的狀態(tài)。應(yīng)用程序開(kāi)始捕獲到每個(gè)未完成的事務(wù)時(shí)運(yùn)行下面的恢復(fù)操作。你可能還希望定期運(yùn)行恢復(fù)操作,以確保數(shù)據(jù)處于一致?tīng)顟B(tài)。達(dá)成一致?tīng)顟B(tài)所需要的時(shí)間取決于應(yīng)用程序需要多長(zhǎng)時(shí)間恢復(fù)每個(gè)事務(wù)。

回滾:

在某些情況下可能需要“回滾”或“撤消”事務(wù),當(dāng)應(yīng)用程序需要“取消”該事務(wù)時(shí),或者是因?yàn)樗肋h(yuǎn)需要恢復(fù)當(dāng)其中一個(gè)帳戶不存在的情況下,或停止現(xiàn)有的事務(wù)。這里有兩種可能的回滾操作:

1、應(yīng)用事務(wù)(即第三步)之后,你已經(jīng)完全提交事務(wù),你不應(yīng)該回滾事務(wù)。相反,創(chuàng)建一個(gè)新的事務(wù),切換源(源)和目標(biāo)(destination)的值。

2、創(chuàng)建事務(wù)(即第一步)之后,在應(yīng)用事務(wù)(即第三步)之前,使用下面的處理過(guò)程:

設(shè)置事務(wù)狀態(tài)為canceling:

首先設(shè)置事務(wù)狀態(tài)為canceling,使用下面的update()操作:

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

撤銷(xiāo)事務(wù):

使用下面的操作順序從兩個(gè)賬戶中撤銷(xiāo)事務(wù):

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集合的內(nèi)容,應(yīng)該類(lèi)似下面:

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

設(shè)置事務(wù)狀態(tài)為canceled:

最后,使用下面的update()狀態(tài)將事務(wù)狀態(tài)設(shè)置為canceled:

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

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

希望本文所述對(duì)大家MongoDB數(shù)據(jù)庫(kù)程序設(shè)計(jì)有所幫助。

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院
国产美女av在线| 国产在线视频精品视频免费看| 爱福利在线视频| 亚洲精品aaaa| 欧美xxxx黑人又粗又长| 黄色av电影在线播放| 国产一二区在线观看| 精品成人一区二区三区免费视频| 国产伦精品一区二区三区高清版禁| 免费影视观看网站入口| 国产精品自产拍在线网站| 国产特黄在线| gogo在线观看| 国产精品免费视频二三区| 18成年在线观看| 久久国产热视频| 国产高清在线观看| 久热国产在线| av在线播放av| 午夜国产福利在线| 一级二级在线观看| 国产精品人人| 国产精品㊣新片速递bt| 99reav| 最新中文字幕在线| 午夜影院在线免费观看| 国产高清视频在线播放| 久久香蕉一区| 毛片网站在线观看| 性网站在线播放| 国产传媒在线播放| 成人亚洲一区二区三区| 久久av少妇| www.91av| 亚洲电影先锋| 亚洲国产精品区| 97最新国自产拍视频在线完整在线看 | 国产男女猛烈无遮挡免费视频| 精品推荐国产麻豆剧传媒| 中文字幕第一页av| 最新国产在线| 最好2018中文免费视频| 国产精品综合久久久久| 国内自拍视频在线观看| 国产女人在线观看| 天天操天天艹| 国产视频三区| 91在线看片| 夜夜嗨yeyeh| a视频免费看| 开心快乐六月丁香婷婷| www.毛片| 欧美卡一卡二| 最近免费中文字幕大全免费第三页| 成人av小说网| 国产一级大片| 在线观看中文字幕| 免费在线你懂的| 国产无套粉嫩白浆在线2022年| 国产精品剧情一区二区在线观看 | 精品999视频| 日本在线视频www鲁啊鲁| 青青草视频在线免费观看| 国内精品一区视频| 天天av天天爽| 97在线超碰| 狠狠操狠狠色| 国产91久久久久蜜臀青青天草二| 先锋影音av中文字幕| 蜜桃视频中文字幕| 国产成a人亚洲精v品| 免费国产阿v视频在线观看| 天堂在线视频| 日本一二三区视频免费高清| 五月天亚洲激情| 日本在线观看网站| 国产精品视频h| 九九久久久2| 99re6在线视频精品免费| 天堂√中文在线| 国产福利在线播放麻豆| eeuss影院在线| 国产精品视频福利一区二区 | 男人操女人免费网站| 国产福利在线视频| 成年网在线观看免费观看网址| 91中文在线| 国产乱xxⅹxx国语对白| av在线官网| 影音av资源站| 91免费日韩| 91福利在线免费| 免费av不卡在线观看| 在线一区二区三区精品| 国产精品第八页| 综合图区亚洲白拍在线| 久久综合第一页| 黄色国产网站在线播放| 免费高清视频日韩| 九九热在线视频免费观看| 精品国产一区二区三区久久久狼牙 | 国产精品久久在线| 国产在线一二三区| 欧美日韩视频精品一区二区| av中文在线| 九色精品视频在线观看| 日本高清中文字幕在线| 黄色av免费看| 最好2018中文免费视频| 国产一二三在线观看| 国产精品入口麻豆完整版| 国产h在线观看| 中文字幕av高清在线观看| 精品伦理一区二区| 伊人免费在线| 免费电影网站在线视频观看福利| 免费a在线观看| 最新av免费看| 亚洲成人av在线影院| av中文天堂在线| 国产麻豆精品视频一区二区 | 国产一区二区三区福利| 国产经典av| av片在线观看| 超碰在线中文| 国产在线三区| 国产精品久久久久久精| 亚洲第一成人在线视频| 狠狠干天天爱| 精品国产99久久久久久| 激情综合网五月激情| 国产女主播在线观看| 免费三级毛片| 国产一区二区三区福利| av在线不卡免费| 国产亚av手机在线观看| 日p在线观看| 国产福利小视频在线| 免费午夜一级| 在线免费看av| 天堂资源在线中文| 人人干人人插| 国产超碰97| 五月婷婷视频在线观看| 日本成人免费网站| 伊人影院在线视频| 国产精品伦理一区二区三区| 久久精品最新免费国产成人| 四虎成人精品在永久在线观看| 国产日本视频| 日本中文字幕视频在线| 国产中文伊人| 久热精品视频在线播放| 欧美黑人乱大交ⅹxxxxx| av手机免费在线观看| 国产爆初菊在线观看免费视频网站| 非洲黑人最猛性xxxx交| 超碰在线观看免费版| 欧美日韩视频精品一区二区| 四虎成年永久免费网站| 香蕉视频在线观看www| 中文字幕2020第一页| 在线观看中文| h网址在线观看| av中文资源在线| 奇米影视狠狠狠| 牛牛热在线视频| 国产嫩草在线视频| 国产精品久久久久久久牛牛| 国产极品视频| 51成人精品网站| 伊人免费视频| 另类专区欧美| 国产色婷婷在线| 在线三级中文| 美女被人操视频在线观看| 久久av少妇| 国产日本在线| 四虎影视成人永久免费观看视频| www在线视频| 国产成人福利| 黄色一级片视频| 国产经典av| 日本视频在线观看一区二区三区| 国产中文在线视频| 国产日本在线视频| 中文字幕2019第三页| 免费三级毛片| 国产三级在线免费| 青青免费在线视频| 国产三级在线免费观看| 99福利在线| 天堂资源在线中文| 一个人看的www免费观看视频| 欧美激情福利视频在线观看免费| 中文字幕在线免费看| 日本动漫理论片在线观看网站| 国产精品久久精品牛牛影视| 青青草视频在线免费观看| 国产黄网站在线观看|