Mybatis-plus学习(六)——MybatiPlus的乐观锁和自动填充功能详解

发布于 2020-12-05  3293 次阅读


7.乐观锁

7.1OptimisticLockerInnerInterceptor

当要更新一条记录的时候,希望这条记录没有被别人更新
乐观锁实现方式:

  • 取出记录时,获取当前version
  • 更新时,带上这个version
  • 执行更新时, set version = newVersion where version = oldVersion
  • 如果version不对,就更新失败

7.2使用方法

字段上加上@Version注解

@Version
private Integer version;

说明:

  • 支持的数据类型只有:int,Integer,long,Long,Date,Timestamp,LocalDateTime
  • 整数类型下 newVersion = oldVersion + 1
  • newVersion 会回写到 entity
  • 仅支持 updateById(id)update(entity, wrapper) 方法
  • update(entity, wrapper) 方法下, wrapper 不能复用!!!

8.自动填充功能

8.1原理

①实现元对象处理器接口:com.baomidou.mybatisplus.core.handlers.MetaObjectHandler

②注解填充字段 @TableField(.. fill = FieldFill.INSERT) 生成器策略部分也可以配置!

public class User {

    // 注意!这里需要标记为填充字段
    @TableField(.. fill = FieldFill.INSERT)
    private String fillField;

    ....
}

③自定义实现类 MyMetaObjectHandler

@Slf4j
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {

    @Override
    public void insertFill(MetaObject metaObject) {
        log.info("start insert fill ....");
        this.strictInsertFill(metaObject, "createTime", LocalDateTime.class, LocalDateTime.now()); // 起始版本 3.3.0(推荐使用)
        // 或者
        this.strictUpdateFill(metaObject, "createTime", () -> LocalDateTime.now(), LocalDateTime.class); // 起始版本 3.3.3(推荐)
        // 或者
        this.fillStrategy(metaObject, "createTime", LocalDateTime.now()); // 也可以使用(3.3.0 该方法有bug)
    }

    @Override
    public void updateFill(MetaObject metaObject) {
        log.info("start update fill ....");
        this.strictUpdateFill(metaObject, "updateTime", LocalDateTime.class, LocalDateTime.now()); // 起始版本 3.3.0(推荐)
        // 或者
        this.strictUpdateFill(metaObject, "updateTime", () -> LocalDateTime.now(), LocalDateTime.class); // 起始版本 3.3.3(推荐)
        // 或者
        this.fillStrategy(metaObject, "updateTime", LocalDateTime.now()); // 也可以使用(3.3.0 该方法有bug)
    }
}

注意事项:

  • 填充原理是直接给entity的属性设置值!!!
  • 注解则是指定该属性在对应情况下必有值,如果无值则入库会是null
  • MetaObjectHandler提供的默认方法的策略均为:如果属性有值则不覆盖,如果填充值为null则不填充
  • 字段必须声明TableField注解,属性fill选择对应策略,该声明告知Mybatis-Plus需要预留注入SQL字段
  • 填充处理器MyMetaObjectHandler在 Spring Boot 中需要声明@Component@Bean注入
  • 要想根据注解FieldFill.xxx字段名以及字段类型来区分必须使用父类的strictInsertFill或者strictUpdateFill方法
  • 不需要根据任何来区分可以使用父类的fillStrategy方法
public enum FieldFill {
    /**
     * 默认不处理
     */
    DEFAULT,
    /**
     * 插入填充字段
     */
    INSERT,
    /**
     * 更新填充字段
     */
    UPDATE,
    /**
     * 插入和更新填充字段
     */
    INSERT_UPDATE
}

8.2在springboot中实现

8.2.1首先在account表中添加createTime和updateTime字段

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-vcRYnImm-1607188361050)(E:\新技术学习\java_study\mybatis-plus学习\images\image-20201125232450730.png)]

8.2.2在Account.java中添加字段以及注释

  //插入时自动生成
    @TableField(fill = FieldFill.INSERT)
    private LocalDateTime createTime;
    //更新时自动生成
    @TableField(fill = FieldFill.INSERT_UPDATE)
    private LocalDateTime updateTime;

8.2.3在Component包中新建一个类AccountMetaHandler.java

package com.moyisuiying.booksystem.component;

import com.baomidou.mybatisplus.core.handlers.MetaObjectHandler;
import lombok.extern.slf4j.Slf4j;
import org.apache.ibatis.reflection.MetaObject;
import org.springframework.stereotype.Component;

import java.time.LocalDateTime;

/**
 * Classname:AccountMetaHandler
 *
 * @description:
 * @author: 陌意随影
 * @Date: 2020-11-25 22:56
 * @Version: 1.0
 **/
@Component
@Slf4j
public class AccountMetaHandler  implements MetaObjectHandler {
    @Override
    public void insertFill(MetaObject metaObject) {
        log.info("start insert fill ....");
        // 起始版本 3.3.0(推荐使用)
        this.strictInsertFill(metaObject, "createTime", LocalDateTime.class, LocalDateTime.now());
    }

    @Override
    public void updateFill(MetaObject metaObject) {
        log.info("start update fill ....");
        // 起始版本 3.3.0(推荐使用)
        this.strictUpdateFill(metaObject, "updateTime", LocalDateTime.class, LocalDateTime.now());
    }
}

8.3测试插入和更新操作

 @Test
    public void testInsert(){
        Account account = new Account();
        account.setName("硝酸钠");
        account.setPassword("ab");
       int fla = accountDao.insert(account);
       System.out.println(fla);

   }
@Test
    public void testUpdate(){
        Account account = new Account();
        account.setId(27);
        account.setName("李四");
        account.setPassword("aaaaa");
       int update = accountDao.updateById(account);
       System.out.println(update);
   }

经过测试可以发现createTime在插入记录时会自动填充数据库中createTime的值;当update更新操作时updateTime会被自动更新。

9.测试的源代码已上传到GitHub中:https://github.com/LJF2402901363/java_study.git

在这里插入图片描述


繁华落尽,雪花漫天飞舞。