MybatisPlus-02

MybatisPlus-02

自动填充

在实际项目中的表会和我们的orders表一样,有更新时间创建时间创建人,更新人等字段

我们可以使用 @TableFieldfill 属性来设置字段的自动填充,让我们能更方便的更新相关字段

示例

①在对应字段上增加注解

使用TableField注解的fill属性来标注哪些字段需要在自动填充,加了注解MP才会在对应的SQL中为我们预留字段,而属性值代表我们在什么进行什么操作时需要预留字段

1
2
3
4
5
6
7
8
9
10
11
/**
* 更新时间
*/
@TableField(fill = FieldFill.INSERT_UPDATE)
private LocalDateTime updateTime;

/**
* 创建时间
*/
@TableField(fill = FieldFill.INSERT)
private LocalDateTime createTime;

②自定义填充处理器 MetaObjectHandler

1
2
3
4
5
6
7
8
9
10
11
12
13
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {
@Override
public void insertFill(MetaObject metaObject) {
this.setFieldValByName("createTime", LocalDateTime.now(), metaObject);
this.setFieldValByName("updateTime", LocalDateTime.now(), metaObject);
}

@Override
public void updateFill(MetaObject metaObject) {
this.setFieldValByName("updateTime", LocalDateTime.now(), metaObject);
}
}

逻辑删除

MP也支持逻辑删除的处理,我们只需要配置好逻辑删除的实体字段名,代表删除的字段值和代表未删除的字段值后即可

注意:如果3.3.0版本之前还需要在对应的字段上加上@TableLogic注解

1
2
3
4
5
6
mybatis-plus:
global-config:
db-config:
logic-delete-field: delFlag # 全局逻辑删除的实体字段名(since 3.3.0,配置后可以忽略不配置步骤2)
logic-delete-value: 1 # 逻辑已删除值(默认为 1)
logic-not-delete-value: 0 # 逻辑未删除值(默认为 0)

乐观锁

并发操作时,我们需要保证对数据的操作不发生冲突,乐观锁就是其中一种方式。

乐观锁就是先加上不存在并发冲突问题,在进行实际数据操作的时候再检查是否冲突。

我们在使用乐观锁时一般在表中增加一个version列,用来记录我们对每天记录操作的版本,每次对某条记录进行过操作是,对应的版本也需要+1,然后我们在每次要进行更新操作时,先查询对应数据的version值,在执行更新时, set version = 老版本+1 where version = 老版本

如果在查询老版本号到更新操作的中间时刻有其他人更新了这条数据,这样这次更新语句就会更新失败

这里在更新时对version的操作如果有我们自己做就会显的有点麻烦,所以MP提供了乐观锁插件

使用后我们就可以非常方便的实现对version的操作

使用

①配置对应插件

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
@Configuration
public class MybatisPlusConfig {
/**
* 旧版
*/
@Bean
public OptimisticLockerInterceptor optimisticLockerInterceptor() {
return new OptimisticLockerInterceptor();
}

/**
* 新版
*/
@Bean
public MybatisPlusInterceptor mybatisPlusInterceptor() {
MybatisPlusInterceptor mybatisPlusInterceptor = new MybatisPlusInterceptor();
mybatisPlusInterceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor());
return mybatisPlusInterceptor;
}
}

②在实体类的字段上加上@Version注解

1
2
@Version
private Integer version;

③更新

注意:在更新前我们一定要先查询到version设置到实体类上再进行更新才能生效

1
2
3
4
5
6
7
8
9
10
11
12
@Test
public void testVersion(){
//查询id为3的数据
QueryWrapper<Orders> queryWrapper = new QueryWrapper<>();
queryWrapper.eq("id",3);
Orders orders = ordersMapper.selectOne(queryWrapper);


//对id为3的数据进行更新 把price修改为88
orders.setPrice(88);
ordersMapper.updateById(orders);
}

这种情况下我们可以看到执行的sql已经发生了变化。

1
2
==>  Preparing: UPDATE orders SET price=?, update_time=?, version=? WHERE id=? AND version=? AND del_flag=0
==> Parameters: 8888(Integer), null, 2(Integer), 2(Long), 1(Integer)

多插件配置问题

我们在使用3.4.0版本以后的MP时,如果需要用到多个插件的话要注意

在配置的时候只需要注入一个MybatisPlusInterceptor 对象,把插件对象添加到MybatisPlusInterceptor 对象中即可

例如:

1
2
3
4
5
6
7
@Bean
public MybatisPlusInterceptor mybatisPlusInterceptor() {
MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));
interceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor());
return interceptor;
}