欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

mybatis?plus樂觀鎖及實現(xiàn)詳解

 更新時間:2022年05月27日 11:37:04   作者:把蘋果咬哭的測試筆記  
這篇文章主要為大家介紹了mybatis?plus樂觀鎖及實現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

樂觀鎖

樂觀鎖”這個詞以前我也沒聽過。上次在測試需求的時候,查詢數(shù)據(jù)庫發(fā)現(xiàn)有一個version字段,于是請教開發(fā)這個字干嘛使

人家回復我:樂觀鎖,解決并發(fā)更新用的。當時大家都忙,咱也不敢多問。

今天就來折騰一下“樂觀鎖”。

一、什么是樂觀鎖

樂觀鎖其實用一句話來形容其作用就是:當要更新一條記錄的時候,希望這條記錄沒有被別人更新,從而實現(xiàn)線程安全的數(shù)據(jù)更新。

結(jié)合下場景,記得那是一張庫存表,有一個字段記錄商品庫存,涉及多個地方都有可能去更新它:

  • 程序A 查詢到了這條數(shù)據(jù),得到庫存是800,準備+200更新成1000,但是還沒更新。
  • 程序B 也查詢到了這條數(shù)據(jù),得到庫存是800,準備-200更新成600,并且提交更新了。

那么,這時候A再提交更新之后,B就會發(fā)現(xiàn)明明是自己是800-200=600,怎么最后變成了1000?
這就是因為A的事務(wù)導致了B的數(shù)據(jù)更新丟失。

文字可能讀起來比較晦澀,有請靈魂畫手:

正常情況下:

  • 按先后順序是, A先更新成1000,然后B再拿1000-200,更新成800,這樣B就沒異議了。
  • 或者實在要2個同時更新,那也只能有一個成功,這樣也沒異議。

二、MP來實現(xiàn)樂觀鎖

樂觀鎖的實現(xiàn),通過增加一個字段,比如version,來記錄每次的更新。
查詢數(shù)據(jù)的時候帶出version的值,執(zhí)行更新的時候,會再去比較version,如果不一致,就更新失敗。

還是用之前的user表,增加了新的字段version。

1.在實體類里增加對于的字段,并且加上自動填充(你也可以每次手動填充)

@Data
public class User {
    @TableId(type = IdType.ID_WORKER)
    private Long id;
    private String name;
    private Integer age;
    private String email;
    @TableField(fill = FieldFill.INSERT)        // 新增的時候填充數(shù)據(jù)
    private Date createTime;
    @TableField(fill = FieldFill.INSERT_UPDATE) // 新增或修改的時候填充數(shù)據(jù)
    private Date updateTime;
    @TableField(fill = FieldFill.INSERT)
    @Version
    private Integer version; // 版本號
}
@Component //此注解表示 將其交給spring去管理
public class MyMetaObjectHandler implements MetaObjectHandler {
    @Override
    public void insertFill(MetaObject metaObject) {
        this.setFieldValByName("createTime", new Date(), metaObject);
        this.setFieldValByName("updateTime", new Date(), metaObject);
        this.setFieldValByName("version", 0, metaObject); //新增就設(shè)置版本值為0
    }
    @Override
    public void updateFill(MetaObject metaObject) {
        this.setFieldValByName("updateTime", new Date(), metaObject);
    }
}

2. 配置插件

為了便于管理,可以見一個包,用于存放各種配置類,順便把配置在啟動類里的mapper掃描也換到這里來。

package com.pingguo.mpdemo.config;
import com.baomidou.mybatisplus.extension.plugins.OptimisticLockerInterceptor;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
// 配置掃描mapper的路徑
@MapperScan("com.pingguo.mpdemo.mapper")
public class MpConfig {
    // 樂觀鎖插件
    @Bean
    public OptimisticLockerInterceptor optimisticLockerInterceptor() {
        return new OptimisticLockerInterceptor();
    }
}

3.測試樂觀鎖

先新增一條測試數(shù)據(jù):

    //    新增
    @Test
    void addUser() {
        User user = new User();
        user.setName("大周");
        user.setAge(22);
        user.setEmail("laowang@123.com");
        userMapper.insert(user);
    }

新增成功,可以看到version值是0。

再來試一下正常的修改:

//      測試樂觀鎖
    @Test
    void testOptimisticLocker() {
        User user = userMapper.selectById(1342502561945915393L);
        user.setName("大周2");
        userMapper.updateById(user);
    }

修改成功,可以看到version 變成了1。

最后,模擬下并發(fā)更新,樂觀鎖更新失敗的情況:

    //  測試樂觀鎖-失敗
    @Test
    void testOptimisticLockerFailed() {
        User user = userMapper.selectById(1342502561945915393L);
        user.setName("大周3");
        User user2 = userMapper.selectById(1342502561945915393L);
        user2.setName("大周4");
        userMapper.updateById(user2); // 這里user2插隊到user前面,先去更新
        userMapper.updateById(user); // 這里由于user2先做了更新后,版本號不對,所以更新失敗
    }

按照樂觀鎖的原理,user2是可以更新成功的,也就是name會修改為“大周4”,version會加1。user因為前后拿到的版本號不對,更新失敗。

結(jié)果符合預期,我們也可以看下mybatis的日志,進一步了解一下:

可以看到上面首先是2個查詢,查詢到的version都是1。

接著,第一個執(zhí)行update語句的時候,where條件中version=1,可以找到數(shù)據(jù),于是更新成功,切更新version=2。

ps:這里圖丟了一個我重新補的一個數(shù)據(jù),說明下意思,忽略ID與上面的不一致。

而第二個再執(zhí)行update的時候,where條件version=1,已經(jīng)找不到了,因為version已經(jīng)被上面的更新成了2,所以更新失敗。

以上就是mybatis plus樂觀鎖及實現(xiàn)詳解的詳細內(nèi)容,更多關(guān)于mybatis plus樂觀鎖的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • SpringBoot?Security使用MySQL實現(xiàn)驗證與權(quán)限管理

    SpringBoot?Security使用MySQL實現(xiàn)驗證與權(quán)限管理

    安全管理是軟件系統(tǒng)必不可少的的功能。根據(jù)經(jīng)典的“墨菲定律”——凡是可能,總會發(fā)生。如果系統(tǒng)存在安全隱患,最終必然會出現(xiàn)問題,這篇文章主要介紹了SpringBoot安全管理Spring?Security基本配置
    2022-11-11
  • java Tapestry4.1.2入門說明教程

    java Tapestry4.1.2入門說明教程

    不必關(guān)心鏈接!不必關(guān)心請求(http request)到了哪里!不必關(guān)心響應(yīng)(http response)要轉(zhuǎn)向哪里!Tapestry構(gòu)建于底層的request-resonse模式,基于Servlet技術(shù),抽象出面向組件開發(fā)的模型。Tapestry關(guān)心的是:頁面、組件、事件、對象、方法、屬性!
    2008-11-11
  • 帶你了解Spring AOP的使用詳解

    帶你了解Spring AOP的使用詳解

    這篇文章主要介紹了Spring AOP的使用詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-07-07
  • Java處理延時任務(wù)的常用幾種解決方案

    Java處理延時任務(wù)的常用幾種解決方案

    本文主要介紹了Java處理延時任務(wù)的常用幾種解決方案,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-05-05
  • 使用Criteria進行分組求和、排序、模糊查詢的實例

    使用Criteria進行分組求和、排序、模糊查詢的實例

    這篇文章主要介紹了使用Criteria進行分組求和、排序、模糊查詢的實例,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • MyBatis-Plus工具使用之EntityWrapper解析

    MyBatis-Plus工具使用之EntityWrapper解析

    這篇文章主要介紹了MyBatis-Plus工具使用之EntityWrapper解析,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • Java畢業(yè)設(shè)計實戰(zhàn)之教室預訂管理系統(tǒng)的實現(xiàn)

    Java畢業(yè)設(shè)計實戰(zhàn)之教室預訂管理系統(tǒng)的實現(xiàn)

    這是一個使用了java+SpringBoot+Maven+Vue+mysql開發(fā)的教室預訂管理系統(tǒng),是一個畢業(yè)設(shè)計的實戰(zhàn)練習,具有教室預訂管理該有的所有功能,感興趣的朋友快來看看吧
    2022-02-02
  • 在IDEA中搭建最小可用SpringMVC項目(純Java配置)

    在IDEA中搭建最小可用SpringMVC項目(純Java配置)

    這篇文章主要介紹了在IDEA中搭建最小可用SpringMVC項目(純Java配置),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-12-12
  • Spring整合Dubbo框架過程及原理解析

    Spring整合Dubbo框架過程及原理解析

    這篇文章主要介紹了Spring整合Dubbo框架過程及原理解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-12-12
  • 如何利用Jackson序列化忽略指定類型的屬性詳解

    如何利用Jackson序列化忽略指定類型的屬性詳解

    這篇文章主要給大家介紹了關(guān)于如何利用Jackson序列化忽略指定類型的屬性,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2018-11-11

最新評論