溫馨提示×

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

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

利用MyBatis-Plus怎么實(shí)現(xiàn)一個(gè)樂(lè)觀鎖更新功能

發(fā)布時(shí)間:2021-01-11 14:54:57 來(lái)源:億速云 閱讀:352 作者:Leah 欄目:開(kāi)發(fā)技術(shù)

今天就跟大家聊聊有關(guān)利用MyBatis-Plus怎么實(shí)現(xiàn)一個(gè)樂(lè)觀鎖更新功能,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

實(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ù)類型只有: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í)間戳類型。

多說(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ù)。

看完上述內(nèi)容,你們對(duì)利用MyBatis-Plus怎么實(shí)現(xiàn)一個(gè)樂(lè)觀鎖更新功能有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI