MongoDB 事務(wù)支持詳解
事務(wù)簡介
事務(wù)是數(shù)據(jù)庫中處理的邏輯單元,每個事務(wù)中包括一個或多個數(shù)據(jù)庫操作,既可以是讀操作,也可以是寫操作。
ACID 是一個“真正”事務(wù)所需要具備的一組屬性集合,指的是原子性(Atomicity)、一致性(Consistency)、隔離性(Isolation)和持久性(Durability)。
原子性指的是,事務(wù)中的所有操作要么都被應(yīng)用,要么都不被應(yīng)用。
一致性指的是,如果數(shù)據(jù)庫在執(zhí)行事務(wù)之前是一致性狀態(tài),那么在事務(wù)執(zhí)行之后,無論事務(wù)是否成功,數(shù)據(jù)庫也應(yīng)該是一致性狀態(tài)。
隔離性指的是,即使數(shù)據(jù)庫中有多個事務(wù)并發(fā)地執(zhí)行,各個事務(wù)之間也不會互相影響,并且在并發(fā)狀態(tài)下執(zhí)行的事務(wù)和串行執(zhí)行的事務(wù)產(chǎn)生的結(jié)果完全相同。
持久性指的是,在事務(wù)成功提交了之后,事務(wù)所變更的數(shù)據(jù)一定會保存起來,而不會因為任何故障導(dǎo)致數(shù)據(jù)丟失。
當(dāng)數(shù)據(jù)庫滿足所有這些屬性,并且只有成功的事務(wù)才會被處理時,它就被稱為是符合 ACID 的數(shù)據(jù)庫。
如何使用事務(wù)
事務(wù)語法
MongoDB 提供了兩種 API 來使用事務(wù):
- 核心 API,與關(guān)系數(shù)據(jù)庫類似的語法(如
start_transaction
和commit_transaction
) - 回調(diào) API,這是使用事務(wù)的推薦方法
核心 API 不為大多數(shù)錯誤提供重試邏輯,它要求開發(fā)人員為操作、事務(wù)提交函數(shù)以及所需的任何重試和錯誤邏輯手動編寫代碼。
與核心 API 不同,回調(diào) API 提供了一個單獨的函數(shù),該函數(shù)封裝了大量功能,包括啟動與指定邏輯會話關(guān)聯(lián)的事務(wù)、執(zhí)行作為回調(diào)函數(shù)提供的函數(shù)以及提交事務(wù)(或在出現(xiàn)錯誤時中止)。此函數(shù)還包含了處理提交錯誤的重試邏輯。
在 MongoDB 4.2 中添加回調(diào) API 是為了簡化使用事務(wù)的應(yīng)用程序開發(fā),也便于添加處理事務(wù)錯誤的應(yīng)用程序重試邏輯。
API 區(qū)別
核心 API | 回調(diào) API |
---|---|
需要顯式調(diào)用才能啟動和提交事務(wù) | 啟動事務(wù)、執(zhí)行指定的操作,然后提交(或在發(fā)生錯誤時終止) |
不包含 TransientTransactionError 和 UnknownTransactionCommitResult 的錯誤處理邏輯,而是提供了為這些錯誤進行自定義處理的靈活性 | 自動為 TransientTransactionError 和 UnknownTransactionCommitResult 提供錯誤處理邏輯 |
要求為特定事務(wù)將顯式的邏輯會話傳遞給 API | 要求為特定事務(wù)將顯式的邏輯會話傳遞給 API |
實際使用
在一個 Python 的例子當(dāng)中,使用核心 API 的偽代碼如下展示:
from pymongo import MongoClient from pymongo.errors import ConnectionFailure, OperationFailure # 顯式開啟一個事務(wù)會話 with client.start_session() as session: while True: try: with session.start_transaction(): # 執(zhí)行事務(wù)中的多個寫操作 pass # 提交事務(wù) session.commit_transaction() except (ConnectionFailure, OperationFailure) as e: # 錯誤處理 if e.has_error_label("UnknownTransactionCommitResult"): # 出現(xiàn)暫時性錯誤,則重試整個事務(wù) continue else: raise
使用核心 API 需要注意錯誤的捕捉和處理,而回調(diào) API 就不需要注意這些,其偽代碼如下展示:
from pymongo import MongoClient def session_callback(session): # 執(zhí)行事務(wù)中的多個寫操作 pass # 顯式開啟一個事務(wù)會話 with client.start_session() as session: session.with_transaction(session_callback)
事務(wù)調(diào)優(yōu)
在使用事務(wù)時,有幾個重要的參數(shù)需要注意??梢詫λ鼈冞M行調(diào)整,以確保應(yīng)用程序能夠最佳地使用事務(wù)。
在 MongoDB 事務(wù)中有兩類主要的限制:
- 第一類與事務(wù)的時間限制有關(guān),控制特定事務(wù)可以運行多長時間、事務(wù)等待獲取鎖的時間以及所有事務(wù)將運行的最大長度
- 第二類與 MongoDB 的 oplog 條目和單個條目的大小限制有關(guān)
時間限制
事務(wù)的默認(rèn)最大運行時間是 1 分鐘。
可以通過在 mongod 實例級別上修改 transactionLifetimeLimitSeconds
的限制來增加。對于分片集群,必須在所有分片副本集成員上設(shè)置該參數(shù)。超過此時間后,事務(wù)將被視為已過期,并由定期運行的清理進程中止。清理進程每 60 秒或每 transactionLifetimeLimitSeconds/2
運行一次,以較小的值為準(zhǔn)。
要顯式設(shè)置事務(wù)的時間限制,建議在提交事務(wù)時指定 maxTimeMS
參數(shù)。實際上會使用 maxTimeMS
和 transactionLifetimeLimitSeconds
中的更小值。
事務(wù)等待獲取其操作所需鎖的默認(rèn)最大時間是 5 毫秒??梢酝ㄟ^修改由 maxTransactionLockRequestTimeoutMillis
參數(shù)控制的限制來增加。如果事務(wù)在此期間無法獲得鎖,則該事務(wù)會被中止。
當(dāng) maxTransactionLockRequestTimeoutMillis
設(shè)置為 0
時,意味著如果事務(wù)無法立即獲得所需的所有鎖,則該事務(wù)會被中止。設(shè)置為 -1
將使用由 maxTimeMS
參數(shù)所指定的特定于操作的超時時間。
oplog 大小限制
MongoDB 會創(chuàng)建出與事務(wù)中寫操作數(shù)量相同的 oplog 條目。
但是,每個 oplog 條目必須在 16MB 的 BSON 文檔大小限制之內(nèi)。
到此這篇關(guān)于MongoDB 事務(wù)支持詳解的文章就介紹到這了,更多相關(guān)MongoDB事務(wù)支持內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
在mac系統(tǒng)下安裝與配置mongoDB數(shù)據(jù)庫
這篇文章主要介紹了在mac系統(tǒng)下安裝與配置mongoDB數(shù)據(jù)庫的操作步驟,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2021-09-09MongoDB 數(shù)據(jù)庫的命名、設(shè)計規(guī)范詳解
隨著MongoDB的普及和使用量的快速增長,為了規(guī)范使用,便于管理和獲取更高的性能,整理此文檔2020-02-02Mongodb使用$pop刪除數(shù)組中元素的操作指南
本文描述怎樣從Mongodb的文檔數(shù)組字段中,使用$pop刪除數(shù)組中的元素,文中通過代碼示例給大家講解的非常詳細,對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-06-06Mongodb如何使用killCursors停止運行的cursor
MongoDB分批向用戶返回數(shù)據(jù)結(jié)果,通過游標(biāo)的移動, mongodb確定當(dāng)前返回結(jié)果的位置,是否要加載更多數(shù)據(jù)到內(nèi)存當(dāng)中,這篇文章主要介紹了Mongodb如何使用killCursors停止運行的cursor,需要的朋友可以參考下2023-12-12mongodb使用docker搭建replicaSet集群與變更監(jiān)聽(最新推薦)
replicaSet和cluster從部署難度相比,replicaSet要簡單許多。如果所存儲的數(shù)據(jù)量規(guī)模不算太大的情況下,那么使用replicaSet方式部署mongodb是一個不錯的選擇,這篇文章主要介紹了mongodb使用docker搭建replicaSet集群與變更監(jiān)聽,需要的朋友可以參考下2023-03-03