MP(MyBatis-Plus)實(shí)現(xiàn)樂(lè)觀鎖更新功能的示例代碼
實(shí)現(xiàn)步驟
step1:添加樂(lè)觀鎖攔截器
MP的其他攔截器功能可以參考官網(wǎng)
@Bean
public MybatisPlusInterceptor mybatisPlusInterceptor() {
MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
interceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor());
return interceptor;
}
step2:配置Entity
@TableField(fill = FieldFill.UPDATE) @Version private Date updateTime;
用更新字段充當(dāng)版本號(hào)。
- 上面的配置需要注意的是:updateTime既配置自動(dòng)填充,又配置了樂(lè)觀鎖功能。MP在進(jìn)行處理時(shí)會(huì)先進(jìn)行樂(lè)觀鎖處理,然后再進(jìn)行自動(dòng)填充。
- 問(wèn)題:前端送了id和一些需要更新的字段過(guò)來(lái),每次需要從數(shù)據(jù)庫(kù)中查出version,然后再進(jìn)行更新(要么前端將版本號(hào)傳過(guò)來(lái));
- 支持的數(shù)據(jù)類(lèi)型只有:int,Integer,long,Long,Date,Timestamp,LocalDateTime;
- 僅支持 updateById(id) 與 update(entity, wrapper) 方法,在 update(entity, wrapper) 方法下, wrapper 不能復(fù)用!!!
- 對(duì)于updateTime這個(gè)字段,在數(shù)據(jù)庫(kù)中建議設(shè)置成時(shí)區(qū)不相關(guān)的時(shí)間戳類(lèi)型。
多說(shuō)一點(diǎn)
使用updateTime作為版本號(hào)可能會(huì)存在一些問(wèn)題。
我們通常需要將updateTime返回給前端頁(yè)面,假如我們不做任何設(shè)置,返回前端的數(shù)據(jù)大概是下面的樣子:
{
"userId": 367,
"address": "上海市自由之路xxxxxx...",
"workUnit": "XXXX",
"createTime": "2020-12-22T00:00:00.000+08:00",
"updateTime": "2021-01-08T17:28:14.782+08:00"
}
這種時(shí)間格式可能不是前端頁(yè)面需要的,這是我們可以進(jìn)行如下設(shè)置;
spring: jackson: default-property-inclusion: non_null time-zone: GMT+8 date-format: yyyy-MM-dd HH:mm:ss
返回的數(shù)據(jù)
{
"userId": 367,
"address": "上海市自由之路xxxxxx...",
"workUnit": "XXXX",
"createTime":"2020-12-22 00:00:00",
"updateTime":"2021-01-08 17:28:14"
}
經(jīng)過(guò)這個(gè)配置后,就可以得到可讀性比較好的時(shí)間格式了。但是我們需要注意的時(shí)候,這個(gè)時(shí)間的精度其實(shí)已經(jīng)丟失了,當(dāng)前提交修改數(shù)據(jù)到后端,這個(gè)值和數(shù)據(jù)庫(kù)中的值已經(jīng)不相等了。所以永遠(yuǎn)不能將數(shù)據(jù)更新成功。
所以這種情況下使用updateTime來(lái)進(jìn)行樂(lè)觀鎖更新就不太適合了。可以考慮在表中另外加一個(gè)字段version來(lái)進(jìn)行樂(lè)觀鎖更新。
但其實(shí)還是有比較好的解決辦法的。
首先,我們不要對(duì)返回的時(shí)間格式進(jìn)行全局話配置。
spring: jackson: default-property-inclusion: non_null time-zone: GMT+8 # date-format: yyyy-MM-dd HH:mm:ss
然后,添加一個(gè)updateTime的備份字段updateTimeSimpleFormat,并對(duì)這個(gè)字段進(jìn)行單獨(dú)的時(shí)間格式化。
private Date updateTime;
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private Date updateTimeSimpleFormat;
updateTimeSimpleFormat不要生成get和set方法,在updateTime的set方法中對(duì)updateTimeSimpleFormat進(jìn)行賦值。
public void setUpdateTime(Date updateTime) {
this.updateTime = updateTime;
this.updateTimeSimpleFormat = updateTime;
}
這樣就既能滿足前端返回格式化的時(shí)間,后端又能獲取到樂(lè)觀鎖的版本號(hào)。
但是,這個(gè)方法比較不好的地方,就是必須對(duì)每個(gè)時(shí)間格式進(jìn)行@JsonFormat注解配置,不能進(jìn)行全局配置,比較繁瑣。
總結(jié):使用updateTime作為樂(lè)觀鎖的優(yōu)點(diǎn)就是不需要再新加字段,比較簡(jiǎn)潔。但是帶來(lái)的問(wèn)題上面已經(jīng)講的很清楚了。還是印證了那個(gè)真理:沒(méi)有完美的技術(shù),只有適合的技術(shù)。
到此這篇關(guān)于MP(MyBatis-Plus)實(shí)現(xiàn)樂(lè)觀鎖更新功能的示例代碼的文章就介紹到這了,更多相關(guān)MyBatis-Plus樂(lè)觀鎖更新內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
idea運(yùn)行java項(xiàng)目main方法報(bào)build failure錯(cuò)誤的解決方法
當(dāng)在使用 IntelliJ IDEA 運(yùn)行 Java 項(xiàng)目的 main 方法時(shí)遇到 "Build Failure" 錯(cuò)誤,這通常意味著在項(xiàng)目的構(gòu)建過(guò)程中遇到了問(wèn)題,以下是一些詳細(xì)的解決步驟,以及一個(gè)簡(jiǎn)單的代碼示例,用于展示如何確保 Java 程序可以成功構(gòu)建和運(yùn)行,需要的朋友可以參考下2024-09-09
Java如何實(shí)現(xiàn)簡(jiǎn)單的RPC框架
這篇文章主要介紹了Java如何實(shí)現(xiàn)簡(jiǎn)單的RPC框架,文中示例代碼非常詳細(xì),幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下2020-07-07
SpringMVC @NotNull校驗(yàn)不生效的解決方案
這篇文章主要介紹了SpringMVC @NotNull校驗(yàn)不生效的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
Java內(nèi)存模型之重排序的相關(guān)知識(shí)總結(jié)
重排序是指編譯器和處理器為了優(yōu)化性能而對(duì)指令序列進(jìn)行重新排序的一種手段,文中詳細(xì)介紹了Java重排序的相關(guān)知識(shí),需要的朋友可以參考下2021-06-06
spring boot攔截器的使用場(chǎng)景示例詳解
這篇文章主要給大家介紹了關(guān)于spring boot攔截器的使用場(chǎng)景,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Spring Boot具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05
java項(xiàng)目怎么集成stable diffusion圖文生成算法
在開(kāi)發(fā)Java項(xiàng)目過(guò)程中,我們經(jīng)常需要使用消息傳遞來(lái)實(shí)現(xiàn)不同組件之間的通信,Stable Diffusion是一種基于消息傳遞的實(shí)時(shí)通信解決方案,使用Java調(diào)用外部服務(wù)(如Python腳本或API服務(wù)),這些服務(wù)運(yùn)行Stable Diffusion模型,本文將介紹如何將Stable Diffusion集成到Java項(xiàng)目2024-07-07

