【mybatis-plus】什么是乐观锁?如何实现“乐观锁”
“樂觀鎖”這個詞以前我也沒聽過。上次在測試需求的時候,查詢數據庫發現有一個version字段,于是請教開發這個字干嘛使,人家回復我:樂觀鎖,結局并發更新用的。
當時大家都忙,咱也不敢多問。
今天就來折騰一下“樂觀鎖”。
一、什么是樂觀鎖
樂觀鎖其實用一句話來形容其作用就是:當要更新一條記錄的時候,希望這條記錄沒有被別人更新,從而實現線程安全的數據更新。
結合下場景,記得那
是一張庫存表,有一個字段記錄商品庫存,涉及多個地方都有可能去更新它:
程序A 查詢到了這條數據,得到庫存是800,準備+200更新成1000,但是還沒更新。
程序B 也查詢到了這條數據,得到庫存是800,準備-200更新成600,并且提交更新了。
那么,這時候A再提交更新之后,B就會發現明明是自己是800-200=600,怎么最后變成了1000?
這就是因為A的事務導致了B的數據更新丟失。
文字可能讀起來比較晦澀,有請靈魂畫手:
正常情況下:
-
按先后順序是, A先更新成1000,然后B再拿1000-200,更新成800,這樣B就沒異議了。
-
或者實在要2個同時更新,那也只能有一個成功,這樣也沒異議。
二、MP來實現樂觀鎖
樂觀鎖的實現,通過增加一個字段,比如version,來記錄每次的更新。
查詢數據的時候帶出version的值,執行更新的時候,會再去比較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)????????//?新增的時候填充數據private?Date?createTime;@TableField(fill?=?FieldFill.INSERT_UPDATE)?//?新增或修改的時候填充數據private?Date?updateTime;@TableField(fill?=?FieldFill.INSERT)@Versionprivate?Integer?version;?//?版本號 } @Component?//此注解表示?將其交給spring去管理 public?class?MyMetaObjectHandler?implements?MetaObjectHandler?{@Overridepublic?void?insertFill(MetaObject?metaObject)?{this.setFieldValByName("createTime",?new?Date(),?metaObject);this.setFieldValByName("updateTime",?new?Date(),?metaObject);this.setFieldValByName("version",?0,?metaObject);?//新增就設置版本值為0}@Overridepublic?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?{//?樂觀鎖插件@Beanpublic?OptimisticLockerInterceptor?optimisticLockerInterceptor()?{return?new?OptimisticLockerInterceptor();} }3.測試樂觀鎖
先新增一條測試數據:
????//????新增@Testvoid?addUser()?{User?user?=?new?User();user.setName("大周");user.setAge(22);user.setEmail("laowang@123.com");userMapper.insert(user);}新增成功,可以看到version值是0。
再來試一下正常的修改:
//??????測試樂觀鎖@Testvoid?testOptimisticLocker()?{User?user?=?userMapper.selectById(1342502561945915393L);user.setName("大周2");userMapper.updateById(user);}修改成功,可以看到version 變成了1。
最后,模擬下并發更新,樂觀鎖更新失敗的情況:
????//??測試樂觀鎖-失敗@Testvoid?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因為前后拿到的版本號不對,更新失敗。
結果符合預期,我們也可以看下mybatis的日志,進一步了解一下:
可以看到上面首先是2個查詢,查詢到的version都是1。
接著,第一個執行update語句的時候,where條件中version=1,可以找到數據,于是更新成功,切更新version=2。
ps:這里圖丟了一個我重新補的一個數據,說明下意思,忽略ID與上面的不一致。
而第二個再執行update的時候,where條件version=1,已經找不到了,因為version已經被上面的更新成了2,所以更新失敗。
總結
以上是生活随笔為你收集整理的【mybatis-plus】什么是乐观锁?如何实现“乐观锁”的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 实战渗透之一个破站日一天
- 下一篇: mmap映射区和shm共享内存的区别总结