【Mybatis-Plus学习笔记三】——Mybatis-Plus实现乐观锁

本文介绍乐观锁的概念及解决并发更新问题的方法,并通过MyBatis-Plus实现乐观锁,包括添加版本字段、配置乐观锁插件及测试过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.乐观锁

  • 乐观锁是在假设数据操作都是安全的,不会出现问题;如果出现了问题,直接返回错误。主要数据并发更新的问题,比如:丢失更新(多个线程同时对某条数据更新,无论执行顺序如何,都会丢失其他线程更新的数据)。

2.MyBatis-plus实现乐观锁

  • 需要在表里增加一个version字段,来记录每次的更新
  • 给verison一个默认值如:0,可以用MyBatis-plus自动填充功能实现
  • 当查询数据时,获取当前version的值
  • 当更新数据时,带上获取的version
  • 当执行更新时,将提交数据的版本数据与数据库表对应记录的当前版本信息进行比对,如果提交的数据版本号等于数据库表当前版本号,则予以更新,否则更新失败。
//使用lombok注解,生成get,set,toString,有参,无参构造等方法
@Data//生成get,set
@AllArgsConstructor//有参构造
@NoArgsConstructor//无参构造
@ToString//toString
public class User{

    @TableId(type = IdType.AUTO)//id自增,数据库也要设置了 ID自增 否则无效
    private Long id;
    private String name;
    private Integer age;
    private String email;

    @TableField(fill = FieldFill.INSERT)//自动填充,新增数据时填充
    private LocalDateTime createTime;
    @TableField(fill = FieldFill.INSERT_UPDATE)//自动填充,新增和更新数据时填充
    private LocalDateTime updateTime;

    @Version//版本号。用于实现乐观锁
    @TableField(fill = FieldFill.INSERT)//自动填充,新增数据时填充
    private Integer version;
}
  • 配置乐观锁插件
@Configuration
public class MyBatisPlusConfig {
    
    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {
        MybatisPlusInterceptor mybatisPlusInterceptor = new MybatisPlusInterceptor();
        mybatisPlusInterceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor());
        return mybatisPlusInterceptor;
    }
}

3.测试

  • 正常情况:新增数据,version自动填充为0;修改数据,version会加1
	@Test
	void add() {
		User user = new User();
		user.setName("小明");
		user.setAge(10);
		int result = userMapper.insert(user);
		if (result > 0) {
			System.out.println("操作成功");
			//操作成功后,id等属性值会回写到user里
			System.out.println(user);
		}
	}

	@Test
	void update() {
		User user = userMapper.selectById(8L);
		user.setAge(30);
		int result = userMapper.updateById(user);
		if (result > 0) {
			System.out.println("操作成功");
			//操作成功后,id等属性值会回写到user里
			System.out.println(user);
		}
	}
  • 模拟并发修改数据测试
	//模拟并发修改数据测试
	@Test
	void testError(){

		//模拟线程1
		User user = userMapper.selectById(8L);
		user.setAge(20);

		//模拟线程2,做了更新操作
		User user2 = userMapper.selectById(8L);
		user2.setAge(100);
		userMapper.updateById(user2);

		//因为模拟线程2先进行更新,改变version,模拟线程1更新失败
		userMapper.updateById(user);//如果没有乐观锁,就会覆盖模拟线程2修改的值
	}

4.demo地址

  • https://download.csdn.net/download/weixin_43817709/19548774
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值