定义数据库表
CREATE TABLE user
(
id BIGINT(20) NOT NULL COMMENT '主键ID',
name VARCHAR(30) NULL DEFAULT NULL COMMENT '姓名',
age INT(11) NULL DEFAULT NULL COMMENT '年龄',
email VARCHAR(50) NULL DEFAULT NULL COMMENT '邮箱',
PRIMARY KEY (id)
);
定义实体类映射
@Data
public class User {
private Long id;
private String name;
private Integer age;
private String email;
}
定义mapper接口,继承MybatiPlus中的BaseMapper
public interface UserMapper extends BaseMapper<User> {
}
利用MybatisPlus实现CRUD
@RunWith(SpringRunner.class)
@SpringBootTest(classes = MybatisPlustApplication.class)
public class MybatisPlustTests {
@Autowired
private UserMapper userMapper;
@Test
public void testSelectList(){
List<User> users = userMapper.selectList(null);
users.forEach(System.out::println);
}
}
查询SQL输出日志的配置
#mybatis日志
mybatis-plus.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl
MyBatis-Plus中的主键策略
- ID_WORKER
mybatis-plus的默认主键生成策略,全局唯一ID
参考资料:分布式系统唯一ID生成方案汇总 - 自增策略
要想实现自增策略的主键增长方式,需要配置如下:- 数据库表中设置为主键自增
- 在实体字段中配置@TableId(type=idType.AUTO)注解
要想一次性影响全局实体的配置,可设置全局主键配置@Data public class User { @TableId(type = IdType.AUTO) private Long id; private String name; private Integer age; private String email; }
其它主键策略:分析 IdType 源码可知#全局设置主键生成策略 mybatis-plus.global-config.db-config.id-type=auto
@Getter
public enum IdType {
/**
* 数据库ID自增
*/
AUTO(0),
/**
* 该类型为未设置主键类型
*/
NONE(1),
/**
* 用户输入ID
* 该类型可以通过自己注册自动填充插件进行填充
*/
INPUT(2),
/* 以下3种类型、只有当插入对象ID 为空,才自动填充。 */
/**
* 全局唯一ID (idWorker)
*/
ID_WORKER(3),
/**
* 全局唯一ID (UUID)
*/
UUID(4),
/**
* 字符串全局唯一ID (idWorker 的字符串表示)
*/
ID_WORKER_STR(5);
private int key;
IdType(int key) {
this.key = key;
}
}
MyBatis-Plus中的自动填充
项目中我们通常会遇到一些数据,每次都使用相同的方式填充,如创建时间,更新时间。
在这时,我们就可以使用自动填充功能来完成对这些数据的自动赋值。
- 在数据库表中添加自动填充字段create_time,update_time
- 在对应的实体映射类中添加相关映射,并添加注解@TableField
@Data
public class User {
@TableId(type = IdType.AUTO)
private Long id;
.......
.......
@TableField(fill = FieldFill.INSERT)
private Date createTime;
@TableField(fill = FieldFill.INSERT_UPDATE)
private Date updateTime;
}
- 实现元对象处理接口
注意:这里需要加上@Component注解将此配置注入到Spring容器中。setFieldValByName()方法中的第一个参数为实体类的属性名称而不是数据库中的字段!
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {
@Override
public void insertFill(MetaObject metaObject) {
this.setFieldValByName("createTime",new Date(),metaObject);
this.setFieldValByName("updateTime",new Date(),metaObject);
}
@Override
public void updateFill(MetaObject metaObject) {
this.setFieldValByName("updateTime",new Date(),metaObject);
}
}
- 此时我们在进行添加操作和更新操作时就不需要给createTime和updateTime字段进行复制,mybatis-plus框架会自动进行字段填充。
==> Preparing: UPDATE user SET name=?, age=?, email=?, update_time=? WHERE id=?
==> Parameters: 李大娃子(String), 20(Integer), 903611454@qq.com(String), 2020-07-10 14:40:54.102(Timestamp), 6(Long)
<== Updates: 1
乐观锁
什么是乐观锁,什么悲观锁,主要解决的是什么问题
当程序中可能出现并发操作时,我们就需要通过使用一定的手段来保证并发情况下数据的准确性,通过这种手段保证,当用户和其他用户一起操作的情况下,所得到的结果和他单独操作时得到的结果是一样的,这种手段就叫做并发控制。并发控制的目的是保证一个用户的操作结果不会对另一个用户的操作产生不和里的影响。
如何没有做好并发控制,就可能导致幻读、脏读和不可重复读
- 主要使用场景:当要更新一条记录是,希望这条记录不被别人更新,也就是说实现线程安全的更新操作。
- mybatis-plus中乐观锁的实现方式:
- 取出记录,获取当前version
- 更新时,带上version
- 执行更新时, set version = newVersion where version = oldVersion
- 如果version不对,就更新失败
- 在数据库表中添加version字段
ALTER TABLE `user` ADD COLUMN `version` INT
- 在实体类中添加version字段,同时添加@Version注解
@Version
@TableField(fill = FieldFill.INSERT)
private Integer version;
- 元对象处理接口添加version的insert默认值
this.setFieldValByName("version",1,metaObject);
特别说明:
- 支持的数据类型只有 int,Integer,long,Long,Date,Timestamp,LocalDateTime
- 整数类型下 newVersion = oldVersion + 1
- newVersion 会回写到 entity 中
- 仅支持 updateById(id) 与 update(entity, wrapper) 方法
- 在 update(entity, wrapper) 方法下, wrapper 不能复用!!!
- 在MybatisPlusConfig类中注册Bean
@Configuration
@EnableTransactionManagement
public class MybatisPlusConfig {
/**
* 乐观锁插件
*/
@Bean
public OptimisticLockerInterceptor optimisticLockerInterceptor(){
return new OptimisticLockerInterceptor();
}
}
- 测试
@Test
public void testOptimisticLocker(){
//根据id取出要被更新的user,注意:此user对象中包含所有属性
User user = userMapper.selectById(7L);
//更新名字
user.setName("李小娃子");
//执行更新
userMapper.updateById(user);
}
==> Preparing: UPDATE user SET name=?, age=?, email=?, create_time=?, update_time=?, version=? WHERE id=? AND version=?
==> Parameters: 李小娃子(String), 20(Integer), 903611454@qq.com(String), 2020-07-10 15:19:46.0(Timestamp), 2020-07-10 15:28:59.518(Timestamp), 2(Integer), 7(Long), 1(Integer)
<== Updates: 1
分页
MyBatis Plus自带分页插件,只要简单的配置即可实现分页功能
- 创建配置类
@Bean
public PaginationInterceptor paginationInterceptor(){
return new PaginationInterceptor();
}
- 测试selectPage分页
@Test
public void selectByPage() {
Page page = new Page(1, 5);
userMapper.selectPage(page, null);
page.getRecords().forEach(System.out::println);
System.out.println(page.getCurrent()); //当前页
System.out.println(page.getSize()); //每页记录数
System.out.println(page.getTotal()); //总记录数
System.out.println(page.getPages()); //总页数
}
@Test
public void testSelectMapsPage(){
Page<User> page = new Page(1,5);
IPage<Map<String, Object>> mapIPage = userMapper.selectMapsPage(page, null);
//注意:此行必须使用 mapIPage 获取记录列表,否则会有数据类型转换错误
mapIPage.getRecords().forEach(System.out::println);
System.out.println(mapIPage.getCurrent()); //当前页
System.out.println(mapIPage.getSize()); //每页记录数
System.out.println(mapIPage.getTotal()); //总记录数
System.out.println(mapIPage.getPages()); //总页数
}
逻辑删除
- 物理删除:真实删除,将对应数据从数据库中删除,之后查询不到此条被删除数据
- 逻辑删除:假删除,将对应数据中代表是否被删除字段状态修改为“被删除状态”,之后在数据库中仍旧能看到此条数据记录
- 在数据库中添加“deleted”字段
ALTER TABLE `user` ADD COLUMN `deleted` boolean
- 在实体类中添加对应映射,并添加@TableLogic注解
@TableLogic
@TableField(fill = FieldFill.INSERT)
private Integer deleted;
- 元对象处理接口添加deleted的insert默认值
this.setFieldValByName("deleted",0,metaObject);
- 在application.properties中添加配置
此为默认值
mybatis-plus.global-config.db-config.logic-delete-value=1
mybatis-plus.global-config.db-config.logic-not-delete-value=0
- 在MybatisPlusConfig中注册Bean
@Bean
public ISqlInjector iSqlInjector(){
return new LogicSqlInjector();
}
性能分析
性能分析拦截器,用于输入每条SQL语句及其执行时间
SQL性能分析器,开发环境使用,超出制定时间,停止运行,有助于发现问题。
- 配置插件
@Bean
@Profile({"dev","test"}) //标注在开发和测试环境中使用
public PerformanceInterceptor performanceInterceptor(){
PerformanceInterceptor performanceInterceptor = new PerformanceInterceptor();
performanceInterceptor.setMaxTime(100); //ms,超出此处设置的ms数,sql语句不执行
performanceInterceptor.setFormat(true); // SQL是否格式化,默认为fasle
return performanceInterceptor;
}