MyBatis-Plus通過(guò)version機(jī)制實(shí)現(xiàn)樂(lè)觀鎖的思路
MyBatis-Plus
是通過(guò)version
機(jī)制實(shí)現(xiàn)樂(lè)觀鎖的。
大致思路:
- 取出記錄,攜帶記錄的當(dāng)前
version
; - 更新記錄的時(shí)候,比較記錄當(dāng)前的
version
是否有改變; - 如果
version
未改變,則更新記錄,并更新version
,一般值+1
; - 如果
version
改變了,則不更新記錄。
version
機(jī)制的核心思想就是,假設(shè)發(fā)生并發(fā)沖突的幾率很低,只有當(dāng)更新數(shù)據(jù)的時(shí)候采取檢查是否有沖突,而判斷是否有沖突的依據(jù)就是version
的值是否被改變了。
配置
MyBatis-Plus
中配置樂(lè)觀鎖分兩步:
- 實(shí)例化
OptimisticLockerInnerInterceptor
,并添加到MyBatis-Plus
的攔截器鏈中; - 定義
version
字段,并在Entity
中使用@Version
注解注釋version
字段。
說(shuō)明:
支持的數(shù)據(jù)類(lèi)型只有:
int
、Integer
、long
、Long
、Date
、Timestamp
、LocalDateTime
;整數(shù)類(lèi)型下
newVersion = oldVersion + 1
;
newVersion
會(huì)回寫(xiě)到entity
中;僅支持
updateById(id)
與update(entity, wrapper)
方法;在
update(entity, wrapper)
方法下,wrapper
不能復(fù)用!!!
配置如下:
首先,實(shí)例化OptimisticLockerInnerInterceptor
,并添加到攔截器鏈中:
@Configuration public class MyBatisPlusConfig { /** * 插件配置 * * @return */ @Bean public MybatisPlusInterceptor mybatisPlusInterceptor() { MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor(); // 向MyBatis-Plus的過(guò)濾器鏈中添加分頁(yè)攔截器,需要設(shè)置數(shù)據(jù)庫(kù)類(lèi)型(主要用于分頁(yè)方言) interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL)); // 添加樂(lè)觀鎖攔截器 interceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor()); return interceptor; } }
然后,使用@Version
注解:
@Data @TableName("tb_user") public class UserEntity { private Long id; private String name; private Integer age; private String email; @TableField(value = "create_time", fill = FieldFill.INSERT) private Date createTime; @TableField(value = "update_time", fill = FieldFill.INSERT_UPDATE) private Date updateTime; @TableLogic(value = "0", delval = "-1") @TableField(value = "delete_flag", fill = FieldFill.INSERT) private Integer deleteFlag; @Version @TableField(value = "version", fill = FieldFill.INSERT) private Integer version; }
配置insert
時(shí)候,version
默認(rèn)值賦1
:
/** * 自動(dòng)填充字段值得配置 */ @Component public class AutoFillFieldValueConfig implements MetaObjectHandler { @Override public void insertFill(MetaObject metaObject) { this.strictInsertFill(metaObject, "createTime", Date.class, new Date()); this.strictInsertFill(metaObject, "updateTime", Date.class, new Date()); this.strictInsertFill(metaObject, "deleteFlag", Integer.class, 0); this.strictInsertFill(metaObject, "version", Integer.class, 1); } @Override public void updateFill(MetaObject metaObject) { this.strictUpdateFill(metaObject, "updateTime", Date.class, new Date()); } }
測(cè)試一下
1.測(cè)試新增記錄
首先新增一條數(shù)據(jù):
@Test public void testVersionInsert() { // 插入一個(gè)新的用戶(hù) UserEntity newUser = new UserEntity(); newUser.setId(12L); newUser.setName("Kelly"); newUser.setAge(28); newUser.setEmail("Kelly@163.com"); userMapper.insert(newUser); }
控制臺(tái)日志:
==> Preparing: INSERT INTO tb_user ( id, name, age, email, create_time, update_time, delete_flag, version ) VALUES ( ?, ?, ?, ?, ?, ?, ?, ? )
==> Parameters: 12(Long), Kelly(String), 28(Integer), Kelly@163.com(String), 2021-09-25 00:14:23.894(Timestamp), 2021-09-25 00:14:23.896(Timestamp), 0(Integer), 1(Integer)
<== Updates: 1
數(shù)據(jù)庫(kù)數(shù)據(jù):
可以看到,新增記錄時(shí),
version
默認(rèn)賦值為1
。
2.測(cè)試更新記錄
下面,來(lái)測(cè)試一下更新記錄,看看version
的變化。
@Test public void testVersionUpdate() { // 查詢(xún)用戶(hù)記錄 UserEntity updateUser = userMapper.selectById(12L); // 更新用戶(hù)記錄 updateUser.setId(12L); updateUser.setAge(30); userMapper.updateById(updateUser); }
注意:這里有一個(gè)坑!
一定要先查詢(xún)出這條數(shù)據(jù),再更新,樂(lè)觀鎖才會(huì)生效?。?!
控制臺(tái)打印的日志:
==> Preparing: SELECT id,name,age,email,create_time,update_time,delete_flag,version FROM tb_user WHERE id=? AND delete_flag=0
==> Parameters: 12(Long)
<== Columns: id, name, age, email, create_time, update_time, delete_flag, version
<== Row: 12, Kelly, 30, Kelly@163.com, 2021-09-25 00:14:24, 2021-09-25 00:20:24, 0, 1
<== Total: 1......
==> Preparing: UPDATE tb_user SET name=?, age=?, email=?, create_time=?, update_time=?, version=? WHERE id=? AND version=? AND delete_flag=0
==> Parameters: Kelly(String), 30(Integer), Kelly@163.com(String), 2021-09-25 00:14:24.0(Timestamp), 2021-09-25 00:20:24.0(Timestamp), 2(Integer), 12(Long), 1(Integer)
<== Updates: 1
數(shù)據(jù)庫(kù)數(shù)據(jù):
可以看到,version
字段由原來(lái)的1
,更新為2
。
到此這篇關(guān)于MyBatis-Plus通過(guò)version機(jī)制實(shí)現(xiàn)樂(lè)觀鎖的思路的文章就介紹到這了,更多相關(guān)MyBatis Plus實(shí)現(xiàn)樂(lè)觀鎖內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring框架學(xué)習(xí)之Spring?@Autowired實(shí)現(xiàn)自動(dòng)裝配的代碼
自動(dòng)裝配就是說(shuō),你不用手動(dòng)實(shí)現(xiàn)bean之間的組合關(guān)系,只要使用了@Autowired注解,程序就會(huì)自動(dòng)的注入這個(gè)需要的bean,前提是你的Spring容器有這個(gè)bean,這篇文章主要介紹了Spring?@Autowired實(shí)現(xiàn)自動(dòng)裝配,需要的朋友可以參考下2021-12-12Java實(shí)現(xiàn)鏈表的常見(jiàn)操作算法詳解
這篇文章主要介紹了Java實(shí)現(xiàn)鏈表的常見(jiàn)操作算法詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09Java基礎(chǔ)之詳細(xì)總結(jié)五種常用運(yùn)算符
在通常代碼邏輯處理中,我們常常都會(huì)使用到運(yùn)算符,今天我們就詳細(xì)了解一下運(yùn)算符的使用以及分類(lèi).運(yùn)算符是對(duì)常量或者變量進(jìn)行操作的符號(hào),它分為算術(shù)運(yùn)算符,賦值運(yùn)算符,比較運(yùn)算符,邏輯運(yùn)算符以及位運(yùn)算符.需要的朋友可以參考下2021-05-05Java中null相關(guān)注解的實(shí)現(xiàn)
本文主要介紹了Java中null相關(guān)注解的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-04-04Java常用的八種排序算法及代碼實(shí)現(xiàn)+圖解
這篇文章主要介紹了Java常用的八種排序算法及代碼實(shí)現(xiàn),在Java的時(shí)候,對(duì)于排序的應(yīng)用需要熟練的掌握,這樣才能夠確保Java學(xué)習(xí)時(shí)候能夠有扎實(shí)的基礎(chǔ)能力。那Java有哪些排序算法呢?本文小編就來(lái)詳細(xì)說(shuō)說(shuō)Java經(jīng)典的8種排序算法,需要的朋友可以參考一下2021-12-12Java模擬登錄正方教務(wù)抓取成績(jī)、課表、空教室
這篇文章主要介紹了Java模擬登錄正方教務(wù)抓取成績(jī)、課表、空教室等信息,Java實(shí)現(xiàn)模擬登錄正方教務(wù)抓取成績(jī)、課表、空教室,通過(guò)HttpClient來(lái)模擬瀏覽器請(qǐng)求,Jsoup解析網(wǎng)頁(yè)內(nèi)容,感興趣的小伙伴們可以參考一下2016-04-04WordPress中卸載插件以及移除文章類(lèi)型組件的代碼示例
這篇文章主要介紹了WordPress中卸載插件以及移除文章類(lèi)型組件的代碼示例,包括卸載函數(shù)鉤子的方法介紹,需要的朋友可以參考下2015-12-12Java中用Socket實(shí)現(xiàn)HTTP文件上傳實(shí)例
本篇文章主要介紹了Java中用Socket實(shí)現(xiàn)HTTP文件上傳實(shí)例,詳細(xì)的介紹了通過(guò)讀取Socket的輸入流來(lái)實(shí)現(xiàn)一個(gè)文件上傳的功能,有興趣的同學(xué)可以一起了解一下2017-04-04