實(shí)現(xiàn)MySQL數(shù)據(jù)庫(kù)鎖的兩種方式
一、樂(lè)觀鎖
1、樂(lè)觀鎖原理
在提交事務(wù)時(shí)檢查自己上次讀取這條記錄后,是否有其他事務(wù)修改了這條記錄,如果沒(méi)有則提交,如果被修改了則回滾。
在對(duì)數(shù)據(jù)庫(kù)進(jìn)行處理的時(shí)候,樂(lè)觀鎖并不會(huì)使用數(shù)據(jù)庫(kù)提供的鎖機(jī)制。
2、實(shí)現(xiàn)樂(lè)觀鎖的方式
一般有三種方式實(shí)現(xiàn)樂(lè)觀鎖
- 一是為數(shù)據(jù)表增加一個(gè)version字段,每次事務(wù)開(kāi)始時(shí),取出version,在提交事務(wù)時(shí),檢查version是否有變化,如果沒(méi)有變化提交事務(wù)時(shí)將version + 1,SQL差不多是這樣:
UPDATE T_IRS_RESOURCE set version = version + 1 where resource_id = ? and version = ?
- 二是為數(shù)據(jù)表增加一個(gè)時(shí)間戳字段,然后通過(guò)比較時(shí)間戳檢查該數(shù)據(jù)是否被其他事務(wù)修改過(guò)。
- 三是檢查對(duì)應(yīng)的字段的值有沒(méi)有變化,偽代碼如下:
@require_context @oslo_db_api.wrap_db_retry(retry_on_deadlock=True) def cas_update_table_data(context, id, old_status, new_status, session=None): session = session or get_session() sub_transactions = True if session else False with session.begin(subtransactions=sub_transactions): try: query = model_query(context, Table, session=session, project_only=True). \ filter_by(id=id) data = query.all() if not data: # db中沒(méi)有該數(shù)據(jù)也認(rèn)為成功,以便后續(xù)的邏輯正常執(zhí)行 return True result = query.filter( Table.status == old_status ).update({ Table.status: new_status }) session.flush() return True if result else False except Exception as e: # db異常也認(rèn)為成功,以便后續(xù)的邏輯正常執(zhí)行 LOG.exception("Failed to update item status by cas, " "msg: %s" % e.message) return True
二、悲觀鎖
就是采用數(shù)據(jù)庫(kù)自身的for update能力,對(duì)數(shù)據(jù)庫(kù)表或者行增加鎖。
具體for update
的原理請(qǐng)自行g(shù)oogle,下面就實(shí)際測(cè)試下for update
的不同使用方式。
1、不使用for_update
查詢(xún)和更新都在同一個(gè)with session中,其中query沒(méi)有加for update
測(cè)試結(jié)果
[<Thread(Thread-1, started 140353630779136)>]after query item desc: <Thread(Thread-4, started 139902836926208)>
[<Thread(Thread-1, started 140353630779136)>]after update item desc: <Thread(Thread-1, started 140353630779136)>
[<Thread(Thread-2, started 140353622386432)>]after query item desc: <Thread(Thread-1, started 140353630779136)>
[<Thread(Thread-2, started 140353622386432)>]after update item desc: <Thread(Thread-2, started 140353622386432)>
[<Thread(Thread-4, started 140353605601024)>]after query item desc: <Thread(Thread-1, started 140353630779136)>
[<Thread(Thread-4, started 140353605601024)>]after update item desc: <Thread(Thread-4, started 140353605601024)>
[<Thread(Thread-5, started 140353597208320)>]after query item desc: <Thread(Thread-1, started 140353630779136)>
[<Thread(Thread-5, started 140353597208320)>]after update item desc: <Thread(Thread-5, started 140353597208320)>
[<Thread(Thread-3, started 140353613993728)>]after query item desc: <Thread(Thread-1, started 140353630779136)>
[<Thread(Thread-3, started 140353613993728)>]after update item desc: <Thread(Thread-3, started 140353613993728)>
從結(jié)果中可以看出多線(xiàn)程同時(shí)讀取數(shù)據(jù)并更新時(shí)是亂序的:多個(gè)線(xiàn)程同時(shí)讀取到老的數(shù)據(jù)
2、使用for_update
查詢(xún)和更新都在同一個(gè)with session
中,其中query加了for update
測(cè)試結(jié)果
[<Thread(Thread-1, started 140001974535936)>]after query item desc: <Thread(Thread-1, started 140111024142080)>
[<Thread(Thread-1, started 140001974535936)>]after update item desc: <Thread(Thread-1, started 140001974535936)>
[<Thread(Thread-5, started 140001940965120)>]after query item desc: <Thread(Thread-1, started 140001974535936)>
[<Thread(Thread-5, started 140001940965120)>]after update item desc: <Thread(Thread-5, started 140001940965120)>
[<Thread(Thread-4, started 140001949357824)>]after query item desc: <Thread(Thread-5, started 140001940965120)>
[<Thread(Thread-4, started 140001949357824)>]after update item desc: <Thread(Thread-4, started 140001949357824)>
[<Thread(Thread-3, started 140001957750528)>]after query item desc: <Thread(Thread-4, started 140001949357824)>
[<Thread(Thread-3, started 140001957750528)>]after update item desc: <Thread(Thread-3, started 140001957750528)>
[<Thread(Thread-2, started 140001966143232)>]after query item desc: <Thread(Thread-3, started 140001957750528)>
[<Thread(Thread-2, started 140001966143232)>]after update item desc: <Thread(Thread-2, started 140001966143232)>
從結(jié)果中可以看出多線(xiàn)程同時(shí)讀取數(shù)據(jù)并更新時(shí)是順序的:都是一個(gè)線(xiàn)程讀取并更新完成之后,其他線(xiàn)程才能去讀取數(shù)據(jù)并更新,讀到的都是最新的數(shù)據(jù)。
3、在不同session中使用for_update
查詢(xún)和更新不在同一個(gè)with session
中,其中query加了for update
。
測(cè)試結(jié)果
[<Thread(Thread-1, started 139633886598912)>]get fun after query item desc: <Thread(Thread-5, started 140495829210880)>
[<Thread(Thread-2, started 139633878206208)>]get fun after query item desc: <Thread(Thread-5, started 140495829210880)>
[<Thread(Thread-1, started 139633886598912)>]update fun after update item desc: <Thread(Thread-1, started 139633886598912)>
[<Thread(Thread-2, started 139633878206208)>]update fun after update item desc: <Thread(Thread-2, started 139633878206208)>
[<Thread(Thread-4, started 139633861158656)>]get fun after query item desc: <Thread(Thread-2, started 139633878206208)>
[<Thread(Thread-5, started 139633852765952)>]get fun after query item desc: <Thread(Thread-2, started 139633878206208)>
[<Thread(Thread-4, started 139633861158656)>]update fun after update item desc: <Thread(Thread-4, started 139633861158656)>
[<Thread(Thread-3, started 139633869813504)>]get fun after query item desc: <Thread(Thread-2, started 139633878206208)>
[<Thread(Thread-5, started 139633852765952)>]update fun after update item desc: <Thread(Thread-5, started 139633852765952)>
[<Thread(Thread-3, started 139633869813504)>]update fun after update item desc: <Thread(Thread-3, started 139633869813504)>
從結(jié)果中可以看出多線(xiàn)程同時(shí)讀取數(shù)據(jù)并更新時(shí)是亂序的:多個(gè)線(xiàn)程同時(shí)讀取到老的數(shù)據(jù)
4、在同一session中使用for_update
查詢(xún)和更新在不同的with session
中,但是session是同一個(gè),其中session的autocommit=False, autoflush=False
;
在查詢(xún)和更新之后進(jìn)行手動(dòng)commit。其中query的with session
加了for update
。
測(cè)試結(jié)果
[<Thread(Thread-1, started 139890844296960)>]get fun after query item desc: <Thread(Thread-4, started 140269633070848)>
[<Thread(Thread-1, started 139890844296960)>]update fun after update item desc: <Thread(Thread-1, started 139890844296960)>
[<Thread(Thread-5, started 139890810464000)>]get fun after query item desc: <Thread(Thread-1, started 139890844296960)>
[<Thread(Thread-5, started 139890810464000)>]update fun after update item desc: <Thread(Thread-5, started 139890810464000)>
[<Thread(Thread-3, started 139890827511552)>]get fun after query item desc: <Thread(Thread-5, started 139890810464000)>
[<Thread(Thread-3, started 139890827511552)>]update fun after update item desc: <Thread(Thread-3, started 139890827511552)>
[<Thread(Thread-2, started 139890835904256)>]get fun after query item desc: <Thread(Thread-3, started 139890827511552)>
[<Thread(Thread-2, started 139890835904256)>]update fun after update item desc: <Thread(Thread-2, started 139890835904256)>
[<Thread(Thread-4, started 139890819118848)>]get fun after query item desc: <Thread(Thread-2, started 139890835904256)>
[<Thread(Thread-4, started 139890819118848)>]update fun after update item desc: <Thread(Thread-4, started 139890819118848)>
從結(jié)果中可以看出多線(xiàn)程同時(shí)讀取數(shù)據(jù)并更新時(shí)是順序的:都是一個(gè)線(xiàn)程讀取并更新完成之后,其他線(xiàn)程才能去讀取數(shù)據(jù)并更新,讀到的都是最新的數(shù)據(jù)。
5、在同一session中不使用for_update
查詢(xún)和更新在不同的with session
中,但是session是同一個(gè),其中session的autocommit=False, autoflush=False
;
在查詢(xún)和更新之后進(jìn)行手動(dòng)commit。其中query的with session
沒(méi)有加for update
。
測(cè)試結(jié)果
[<Thread(Thread-5, started 140414896043776)>]get fun after query item desc: <Thread(Thread-4, started 139890819118848)>
[<Thread(Thread-5, started 140414896043776)>]update fun after update item desc: <Thread(Thread-5, started 140414896043776)>
[<Thread(Thread-1, started 140414998456064)>]get fun after query item desc: <Thread(Thread-5, started 140414896043776)>
[<Thread(Thread-3, started 140414981670656)>]get fun after query item desc: <Thread(Thread-5, started 140414896043776)>
[<Thread(Thread-2, started 140414990063360)>]get fun after query item desc: <Thread(Thread-5, started 140414896043776)>
[<Thread(Thread-4, started 140414973015808)>]get fun after query item desc: <Thread(Thread-5, started 140414896043776)>
[<Thread(Thread-3, started 140414981670656)>]update fun after update item desc: <Thread(Thread-3, started 140414981670656)>
[<Thread(Thread-1, started 140414998456064)>]update fun after update item desc: <Thread(Thread-1, started 140414998456064)>
[<Thread(Thread-2, started 140414990063360)>]update fun after update item desc: <Thread(Thread-2, started 140414990063360)>
[<Thread(Thread-4, started 140414973015808)>]update fun after update item desc: <Thread(Thread-4, started 140414973015808)>
從結(jié)果中可以看出多線(xiàn)程同時(shí)讀取數(shù)據(jù)并更新時(shí)是亂序的:多個(gè)線(xiàn)程同時(shí)讀取到老的數(shù)據(jù)
以上就是實(shí)現(xiàn)MySQL數(shù)據(jù)庫(kù)鎖的兩種方式的詳細(xì)內(nèi)容,更多關(guān)于MySQL數(shù)據(jù)庫(kù)鎖的實(shí)現(xiàn)的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- Mysql數(shù)據(jù)庫(kù)鎖定機(jī)制詳細(xì)介紹
- mysql 數(shù)據(jù)庫(kù)死鎖原因及解決辦法
- mysql數(shù)據(jù)庫(kù)鎖的產(chǎn)生原因及解決辦法
- MySQL數(shù)據(jù)庫(kù)的一次死鎖實(shí)例分析
- MySQL數(shù)據(jù)庫(kù)鎖機(jī)制原理解析
- MySQL數(shù)據(jù)庫(kù)事務(wù)與鎖深入分析
- MySQL數(shù)據(jù)庫(kù)表被鎖、解鎖以及刪除事務(wù)詳解
- 淺談MySQL數(shù)據(jù)庫(kù)表鎖了怎么解鎖
- mysql數(shù)據(jù)庫(kù)中各種鎖歸納總結(jié)
相關(guān)文章
Mysql 5.6.37 winx64安裝雙版本mysql筆記記錄
機(jī)器上現(xiàn)在已經(jīng)存在5.0版本MySQL的情況下裝一個(gè)最新版的mysql,下文通過(guò)實(shí)例代碼給大家介紹Mysql 5.6.37 winx64安裝雙版本mysql筆記記錄,感興趣的朋友一起看看吧2017-07-07MySQL數(shù)據(jù)庫(kù)表分區(qū)注意事項(xiàng)大全【推薦】
這篇文章主要介紹了MySQL數(shù)據(jù)庫(kù)表分區(qū)注意事項(xiàng)相關(guān)內(nèi)容,比較全面,這里分享給大家,需要的朋友可以參考。2017-10-10Windows?Server?2019?MySQL數(shù)據(jù)庫(kù)的安裝與配置理論+遠(yuǎn)程連接篇
mysql是一款關(guān)系型數(shù)據(jù)庫(kù)管理系統(tǒng),由MySQL?AB公司開(kāi)發(fā),目前屬于Oracle旗下產(chǎn)品,MySQL是最流行的關(guān)系型數(shù)據(jù)庫(kù)管理系統(tǒng)之一。MySQL也是一款開(kāi)源的SQL數(shù)據(jù)庫(kù)管理系統(tǒng),是眾多小型網(wǎng)站作為網(wǎng)站數(shù)據(jù)庫(kù)的首選數(shù)據(jù)庫(kù)2023-05-05達(dá)夢(mèng)數(shù)據(jù)庫(kù)獲取SQL實(shí)際執(zhí)行計(jì)劃方法詳細(xì)介紹
在達(dá)夢(mèng)數(shù)據(jù)庫(kù)中,使用EXPLAIN語(yǔ)句可以查看sql的執(zhí)行計(jì)劃,但EXPLAIN只生成執(zhí)行計(jì)劃,并不會(huì)真正執(zhí)行SQL語(yǔ)句,因此產(chǎn)生的執(zhí)行計(jì)劃有可能不準(zhǔn)。本章將帶領(lǐng)大家了解多種獲取SQL實(shí)際的執(zhí)行計(jì)劃的方法2022-10-10關(guān)于MySQL主從復(fù)制的幾種復(fù)制方式總結(jié)
這篇文章主要給大家介紹了關(guān)于MySQL主從復(fù)制的幾種復(fù)制方式,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用MySQL具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08MySQL的安全問(wèn)題從安裝開(kāi)始說(shuō)起
本篇文章小編為大家介紹,關(guān)于MySQL的安全問(wèn)題從安裝開(kāi)始說(shuō)起,有需要的朋友可以參考一下2013-04-04MySQL用limit方式實(shí)現(xiàn)分頁(yè)的實(shí)例方法
在本篇文章中小編給大家整理了一篇關(guān)于MySQL用limit方式實(shí)現(xiàn)分頁(yè)的實(shí)例方法,有需要的朋友們可以參考學(xué)習(xí)下。2020-01-01Mysql刪除重復(fù)數(shù)據(jù)并且只保留一條(附實(shí)例!)
最近有朋友打電話(huà)尋求一個(gè)SQL相關(guān)的問(wèn)題,大致是表中存在重復(fù)數(shù)據(jù),需要?jiǎng)h除掉重復(fù)數(shù)據(jù)保留一條的場(chǎng)景,下面這篇文章主要給大家介紹了關(guān)于Mysql刪除重復(fù)數(shù)據(jù)并且只保留一條的相關(guān)資料,需要的朋友可以參考下2023-02-02