亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

MySQL悲觀鎖與樂觀鎖如何實現

發布時間:2021-11-02 13:49:08 來源:億速云 閱讀:132 作者:小新 欄目:開發技術

這篇文章主要為大家展示了“MySQL悲觀鎖與樂觀鎖如何實現”,內容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領大家一起研究并學習一下“MySQL悲觀鎖與樂觀鎖如何實現”這篇文章吧。

前言

悲觀鎖和樂觀鎖是用來解決并發問題的兩種思想,在不同的平臺有著各自的實現。例如在Java中,synchronized就可以認為是悲觀鎖的實現(不嚴謹,有鎖升級的過程,升級到重量級鎖才算),Atomic***原子類可以認為是樂觀鎖的實現。

悲觀鎖

具有強烈的獨占和排他特性,在整個處理過程中將數據處于鎖定狀態,一般是通過系統的互斥量來實現。當其他線程想要獲取鎖時會被阻塞,直到持有鎖的線程釋放鎖。

樂觀鎖

對數據的修改和訪問持樂觀態度,假設不會發生沖突,只有當數據提交更新時才會對數據沖突與否進行檢測,如果沒有沖突則順利提交更新,否則快速失敗,返回一個錯誤給用戶,讓用戶選擇接下來該如何去做,一般來說失敗后會繼續重試,直到提交更新成功為止。

MySQL本身就支持鎖機制,例如我們有一個「先查再寫」的需求,我們希望整個流程是一個原子操作,中間不能被打斷,這時候就可以通過給查詢的數據行加「排他鎖」來實現。只要當前事務不釋放鎖,其他事務要想獲得排他鎖,MySQL就會將其阻塞,直到當前事務釋放鎖。這種MySQL底層的排他鎖就稱作「悲觀鎖」。

MySQL本身不提供樂觀鎖的功能,需要開發者自己實現。普遍的做法是在表中加一個version列,用來標記數據行的版本,當我們需要更新數據時,必須比對version版本,version一致說明這個期間數據沒有被其他事務修改過,否則說明數據已經被其他事務修改,需要自旋重試了。

實戰

假設數據庫有兩張表:商品表和訂單表。

用戶下單后需要執行兩個操作:

  1. 商品表減去庫存。

  2. 訂單表創建一條記錄。

初始數據:ID為1的商品有100的庫存,訂單表數據為空。

客戶端啟動10個線程并發下單,分別在無鎖、悲觀鎖、樂觀鎖的場景下有哪些表現。

如下是創建表的sql語句:

-- 商品表
CREATE TABLE `goods` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `goods_name` varchar(50) NOT NULL,
  `price` decimal(10,2) NOT NULL,
  `stock` int(11) DEFAULT '0',
  `version` int(10) unsigned NOT NULL DEFAULT '0',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8

-- 訂單表
CREATE TABLE `t_order` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `goods_id` bigint(20) NOT NULL,
  `order_time` datetime NOT NULL,
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8

1、無鎖

不做任何處理。

// 下單
private boolean order(){
    Goods goods = goodsMapper.selectById(1L);
    boolean success = false;
    if (goods.getStock() > 0) {
        goods.setStock(goods.getStock() - 1);
        // 更新庫存
        goodsMapper.updateById(goods);
        // 創建訂單
        orderMapper.save(goods.getId());
        success = true;
    }
    return success;
}

控制臺輸出結果:

MySQL悲觀鎖與樂觀鎖如何實現

2、悲觀鎖

查詢商品時加FOR UPDATE,給數據行加排他鎖,這樣其他線程再查詢時就會被阻塞,直到當前線程的事務提交并釋放鎖,其他線程才能繼續下單。這種方式并發性能不高。

sql語句

@Select("SELECT * FROM goods WHERE id = #{id} FOR UPDATE")
Goods selectForUpdate(Long id);

控制臺輸出結果:

MySQL悲觀鎖與樂觀鎖如何實現

注意:FOR UPDATE必須在事務中才有效,查詢和更新必須在同一個事務中!!!

3、樂觀鎖

實現思路是:每次更新時校驗版本號,如果版本號一致說明期間數據沒有被其他線程改過,當前線程可以正常提交更新,否則說明數據已經被其他線程改過了,當前線程需要自旋重試,直到業務成功為止。

更新數據的同時版本號必須自增!!!

@Update("UPDATE goods SET stock = #{stock},version = version+1 WHERE id = #{id} AND version = #{version}")
int updateByVersion(Long id, Integer stock, Integer version);

業務代碼

boolean order(){
    Goods goods = goodsMapper.selectById(1L);
    boolean success = false;
    if (goods.getStock() > 0) {
        goods.setStock(goods.getStock() - 1);
        // 更新庫存,帶上版本號
        int result = goodsMapper.updateByVersion(goods.getId(), goods.getStock(), goods.getVersion());
        if (result <= 0) {
            // 更新失敗,說明期間數據已經被其他線程修改,需要遞歸重試
            return order();
        }
        // 創建訂單
        orderMapper.save(goods.getId());
        success = true;
    }
    return success;
}

控制臺輸出結果:

MySQL悲觀鎖與樂觀鎖如何實現

以上是“MySQL悲觀鎖與樂觀鎖如何實現”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

桐梓县| 堆龙德庆县| 洱源县| 榆林市| 张北县| 山东| 莫力| 广州市| 临泉县| 濉溪县| 囊谦县| 普安县| 潜山县| 乌恰县| 杭锦旗| 山阳县| 成武县| 景泰县| 邻水| 安新县| 黑龙江省| 阳春市| 尼木县| 成安县| 天台县| 改则县| 科尔| 南溪县| 格尔木市| 大理市| 禄丰县| 南澳县| 南充市| 年辖:市辖区| 三亚市| 安塞县| 庆阳市| 琼海市| 上虞市| 同心县| 龙门县|