使用场景
用于当有多个用户同时修改同一条数据的时候,只允许有一个修改成功。
实现原理
使用一个字段,用于记录数据的版本。
当修改数据时,会去检测当前版本是否是正在修改的版本,同时修改成功后会把 版本号 + 1
。
实现方式
- 配置插件
- 在实体类的字段上加上
@Version
注解
代码
配置插件
package com.example.core.config;
import com.baomidou.mybatisplus.annotation.DbType;
import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.BlockAttackInnerInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.OptimisticLockerInnerInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
@MapperScan("com.example.web")
public class MybatisPlusConfig {
/**
* 添加拦截器
*/
@Bean
public MybatisPlusInterceptor mybatisPlusInterceptor() {
MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
interceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor()); // 乐观锁插件
interceptor.addInnerInterceptor(new BlockAttackInnerInterceptor()); // 针对 update 和 delete 语句 作用: 阻止恶意的全表更新删除
interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));// 如果配置多个插件,切记分页最后添加
// interceptor.addInnerInterceptor(new PaginationInnerInterceptor()); 如果有多数据源可以不配具体类型 否则都建议配上具体的DbType
return interceptor;
}
}
在实体类的字段上加上@Version
注解
package com.example.web.entity;
import com.baomidou.mybatisplus.annotation.Version;
import lombok.Data;
@Data
public class User {
// 其他字段
/**
* 版本
*/
@Version
private Integer version;
}
数据库表
测试
代码
/**
* 插入用户
*/
@Test
public void insert() {
User user = new User();
user.setId(16L);
user.setName("郑一");
user.setAge(30);
user.setEmail("zhengyi@example.com");
user.setGender(GenderEnum.MALE);
mapper.insert(user);
}
/**
* 更新用户:版本号为空,乐观锁失效。
*/
@Test
public void update() {
User user = new User();
user.setId(16L);
user.setAge(31);
mapper.updateById(user);
}
/**
* 更新用户:版本号 +1(版本号不为空,乐观锁有效)。
*/
@Test
public void updateWithVersion() {
User user = mapper.selectById(16L);
user.setAge(31);
mapper.updateById(user);
}
/**
* 更新用户:测试同时更新,第二个更新失败。
*/
@Test
public void updateConcurrent() {
// 同步查询
User user1 = mapper.selectById(16L);
user1.setAge(32);
User user2 = mapper.selectById(16L);
user2.setAge(33);
// 更新
mapper.updateById(user1);
mapper.updateById(user2);
}
新插入的数据
更新时版本号为空,乐观锁失效
感觉这像是一个漏洞。
更新时版本号不为空,乐观锁有效
当 实体 的 version
字段不为空
时,乐观锁才能正常生效。
模拟同时更新
同一条数据,查询两次出来。然后调用两次更新,第一次更新成功,第二次更新失败。
查询两次数据
两次更新,第一次成功了,Updates: 1
;第二次失败了,Updates: 0
。