mysql更新鎖實現代碼,mysql加鎖過程詳解

本文目錄一覽:

MySQL – for update 行鎖 表鎖

for update 的作用是在查詢的時候為行加上排它鎖,當一個事務的操作未完成時候,其他事務可以讀取但是不能寫入或更新。

它的典型使用場景是 高並發並且對於數據的準確性有很高要求 ,比如金錢、庫存等,一般這種操作都是很長一串並且開啟事務的,假如現在要對庫存進行操作,在剛開始讀的時候是1,然後馬上另外一個進程將庫存更新為0了,但事務還沒結束,會一直用1進行後續的邏輯,就會有問題,所以需要用for upate 加鎖防止出錯。

行鎖的具體實現演算法有三種:record lock、gap lock以及next-key lock。

只在可重複讀或以上隔離級別下的特定操作才會取得 gap lock 或 next-key lock,在 Select、Update 和 Delete 時,除了基於唯一索引的查詢之外,其它索引查詢時都會獲取 gap lock 或 next-key lock,即鎖住其掃描的範圍。主鍵索引也屬於唯一索引,所以主鍵索引是不會使用 gap lock 或 next-key lock

for update 僅適用於InnoDB,並且必須開啟事務,在begin與commit之間才生效。

select 語句默認不獲取任何鎖,所以是可以讀被其它事務持有排它鎖的數據的!

InnoDB 既實現了行鎖,也實現了表鎖。

當有明確指定的主鍵/索引時候,是行級鎖,否則是表級鎖

假設表 user,存在有id跟name欄位,id是主鍵,有5條數據。

明確指定主鍵,並且有此記錄,行級鎖

無主鍵/索引,表級鎖

主鍵/索引不明確,表級鎖

明確指定主鍵/索引,若查無此記錄,無鎖

參考博文:

Java如何實現對Mysql資料庫的行鎖

下面通過一個例子來說明

場景如下:

用戶賬戶有餘額,當發生交易時,需要實時更新餘額。這裡如果發生並發問題,那麼會造成用戶餘額和實際交易的不一致,這對公司和客戶來說都是很危險的。

那麼如何避免:

網上查了下,有以下兩種方法:

1、使用悲觀鎖

當需要變更餘額時,通過代碼在事務中對當前需要更新的記錄設置for update行鎖,然後開始正常的查詢和更新操作

這樣,其他的事務只能等待該事務完成後方可操作

當然要特別注意,如果使用了Spring的事務註解,需要配置一下:

!– (事務管理)transaction manager, use JtaTransactionManager for global tx —

bean id=”transactionManager”

class=”org.springframework.jdbc.datasource.DataSourceTransactionManager”

property name=”dataSource” ref=”dataSource” /

/bean

!– 使用annotation定義事務 —

tx:annotation-driven transaction-manager=”transactionManager” /

在指定代碼處添加事務註解

@Transactional

@Override

public boolean increaseBalanceByLock(Long userId, BigDecimal amount)

throws ValidateException {

long time = System.currentTimeMillis();

//獲取對記錄的鎖定

UserBalance balance = userBalanceDao.getLock(userId);

LOGGER.info(“[lock] start. time: {}”, time);

if (null == balance) {

throw new ValidateException(

ValidateErrorCode.ERRORCODE_BALANCE_NOTEXIST,

“user balance is not exist”);

}

boolean result = userBalanceDao.increaseBalanceByLock(balance, amount);

long timeEnd = System.currentTimeMillis();

LOGGER.info(“[lock] end. time: {}”, timeEnd);

return result;

}

MyBatis中的鎖定方式,實際測試該方法確實可以有效控制,不過在大並發量的情況下,可能會有性能問題吧

select id=”getLock” resultMap=”BaseResultMap” parameterType=”java.lang.Long”

![CDATA[

select * from user_balance where id=#{id,jdbcType=BIGINT} for update;

]]

/select

2、使用樂觀鎖

這個方法也同樣可以解決場景中描述的問題(我認為比較適合併不頻繁的操作):

設計表的時候增加一個version(版本控制欄位),每次需要更新餘額的時候,先獲取對象,update的時候根據version和id為條件去更新,如果更新回來的數量為0,說明version已經變更

需要重複一次更新操作,如下:sql腳本

update user_balance set Balance = #{balance,jdbcType=DECIMAL},Version = Version+1 where Id = #{id,jdbcType=BIGINT} and Version = #{version,jdbcType=BIGINT}

這是一種不使用資料庫鎖的方法,解決方式也很巧妙。當然,在大量並發的情況下,一次扣款需要重複多次的操作才能成功,還是有不足之處的。不知道還有沒有更好的方法。

用 MySQL 實現分散式鎖,你聽過嗎?

以前參加過一個庫存系統,由於其業務複雜性,搞了很多個應用來支撐。這樣的話一份庫存數據就有可能同時有多個應用來修改庫存數據。

比如說,有定時任務域xx.cron,和SystemA域和SystemB域這幾個JAVA應用,可能同時修改同一份庫存數據。如果不做協調的話,就會有臟數據出現。

對於跨JAVA進程的線程協調,可以藉助外部環境,例如DB或者Redis。下文介紹一下如何使用DB來實現分散式鎖。

本文設計的分散式鎖的交互方式如下:

在使用synchronized關鍵字的時候,必須指定一個鎖對象。

進程內的線程可以基於obj來實現同步。obj在這裡可以理解為一個鎖對象。如果線程要進入synchronized代碼塊里,必須先持有obj對象上的鎖。這種鎖是JAVA裡面的內置鎖,創建的過程是線程安全的。那麼藉助DB,如何保證創建鎖的過程是線程安全的呢?

可以利用DB中的UNIQUE KEY特性,一旦出現了重複的key,由於UNIQUE KEY的唯一性,會拋出異常的。在JAVA裡面,是 SQLIntegrityConstraintViolationException 異常。

transaction_id是事務Id,比如說,可以用

來組裝一個transaction_id,表示某倉庫某銷售模式下的某個條碼資源。不同條碼,當然就有不同的transaction_id。如果有兩個應用,拿著相同的transaction_id來創建鎖資源的時候,只能有一個應用創建成功。

在寫操作頻繁的業務系統中,通常會進行分庫,以降低單資料庫寫入的壓力,並提高寫操作的吞吐量。如果使用了分庫,那麼業務數據自然也都分配到各個資料庫上了。

在這種水平切分的多資料庫上使用DB分散式鎖,可以自定義一個DataSouce列表。並暴露一個 getConnection(String transactionId) 方法,按照transactionId找到對應的Connection。

實現代碼如下:

首先編寫一個initDataSourceList方法,並利用Spring的PostConstruct註解初始化一個DataSource 列表。相關的DB配置從db.properties讀取。

DataSource使用阿里的DruidDataSource。

接著最重要的一個實現getConnection(String transactionId)方法。實現原理很簡單,獲取transactionId的hashcode,並對DataSource的長度取模即可。

連接池列表設計好後,就可以實現往distributed_lock表插入數據了。

接下來利用DB的 select for update 特性來鎖住線程。當多個線程根據相同的transactionId並發同時操作 select for update 的時候,只有一個線程能成功,其他線程都block住,直到 select for update 成功的線程使用commit操作後,block住的所有線程的其中一個線程才能開始幹活。

我們在上面的DistributedLock類中創建一個lock方法。

當線程執行完任務後,必須手動的執行解鎖操作,之前被鎖住的線程才能繼續幹活。在我們上面的實現中,其實就是獲取到當時 select for update 成功的線程對應的Connection,並實行commit操作即可。

那麼如何獲取到呢?我們可以利用ThreadLocal。首先在DistributedLock類中定義

每次調用lock方法的時候,把Connection放置到ThreadLocal裡面。我們修改lock方法。

這樣子,當獲取到Connection後,將其設置到ThreadLocal中,如果lock方法出現異常,則將其從ThreadLocal中移除掉。

有了這幾步後,我們可以來實現解鎖操作了。我們在DistributedLock添加一個unlock方法。

畢竟是利用DB來實現分散式鎖,對DB還是造成一定的壓力。當時考慮使用DB做分散式的一個重要原因是,我們的應用是後端應用,平時流量不大的,反而關鍵的是要保證庫存數據的正確性。對於像前端庫存系統,比如添加購物車佔用庫存等操作,最好別使用DB來實現分散式鎖了。

如果想鎖住多份數據該怎麼實現?比如說,某個庫存操作,既要修改物理庫存,又要修改虛擬庫存,想鎖住物理庫存的同時,又鎖住虛擬庫存。其實也不是很難,參考lock方法,寫一個multiLock方法,提供多個transactionId的入參,for循環處理就可以了。這個後續有時間再補上。

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-tw/n/153873.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2024-11-15 03:23
下一篇 2024-11-15 03:23

相關推薦

  • Python周杰倫代碼用法介紹

    本文將從多個方面對Python周杰倫代碼進行詳細的闡述。 一、代碼介紹 from urllib.request import urlopen from bs4 import Bea…

    編程 2025-04-29
  • Java Bean載入過程

    Java Bean載入過程涉及到類載入器、反射機制和Java虛擬機的執行過程。在本文中,將從這三個方面詳細闡述Java Bean載入的過程。 一、類載入器 類載入器是Java虛擬機…

    編程 2025-04-29
  • Python字元串寬度不限制怎麼打代碼

    本文將為大家詳細介紹Python字元串寬度不限制時如何打代碼的幾個方面。 一、保持代碼風格的統一 在Python字元串寬度不限制的情況下,我們可以寫出很長很長的一行代碼。但是,為了…

    編程 2025-04-29
  • Python基礎代碼用法介紹

    本文將從多個方面對Python基礎代碼進行解析和詳細闡述,力求讓讀者深刻理解Python基礎代碼。通過本文的學習,相信大家對Python的學習和應用會更加輕鬆和高效。 一、變數和數…

    編程 2025-04-29
  • Python滿天星代碼:讓編程變得更加簡單

    本文將從多個方面詳細闡述Python滿天星代碼,為大家介紹它的優點以及如何在編程中使用。無論是剛剛接觸編程還是資深程序員,都能從中獲得一定的收穫。 一、簡介 Python滿天星代碼…

    編程 2025-04-29
  • 倉庫管理系統代碼設計Python

    這篇文章將詳細探討如何設計一個基於Python的倉庫管理系統。 一、基本需求 在著手設計之前,我們首先需要確定倉庫管理系統的基本需求。 我們可以將需求分為以下幾個方面: 1、庫存管…

    編程 2025-04-29
  • 寫代碼新手教程

    本文將從語言選擇、學習方法、編碼規範以及常見問題解答等多個方面,為編程新手提供實用、簡明的教程。 一、語言選擇 作為編程新手,選擇一門編程語言是很關鍵的一步。以下是幾個有代表性的編…

    編程 2025-04-29
  • Python實現簡易心形代碼

    在這個文章中,我們將會介紹如何用Python語言編寫一個非常簡單的代碼來生成一個心形圖案。我們將會從安裝Python開始介紹,逐步深入了解如何實現這一任務。 一、安裝Python …

    編程 2025-04-29
  • 怎麼寫不影響Python運行的長段代碼

    在Python編程的過程中,我們不可避免地需要編寫一些長段代碼,包括函數、類、複雜的控制語句等等。在編寫這些代碼時,我們需要考慮代碼可讀性、易用性以及對Python運行性能的影響。…

    編程 2025-04-29
  • 北化教務管理系統介紹及開發代碼示例

    本文將從多個方面對北化教務管理系統進行介紹及開發代碼示例,幫助開發者更好地理解和應用該系統。 一、項目介紹 北化教務管理系統是一款針對高校學生和教職工的綜合信息管理系統。系統實現的…

    編程 2025-04-29

發表回復

登錄後才能評論