Mybatis-Plus(简称MP)是一个Mybatis的增强工具,在Mybatis的基础上只做增强不做改变,为简化开发、提升效率而生。就像魂斗罗的1P、2P,基友搭配,效率翻倍。
官网:https://mp.baomidou.com/
DROP TABLE IF EXISTS user;
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)
);
2. 对应的Data脚本
DELETE FROM user
INSERT INTO user (id, name, age, email) VALUES
(1, 'Jone', 18, 'test1@baomidou.com'),
(2, 'Jack', 20, 'test2@baomidou.com'),
(3, 'Tom', 28, 'test3@baomidou.com'),
(4, 'Sandy', 21, 'test4@baomidou.com'),
(5, 'Billie', 24, 'test5@baomidou.com');
使用Spring Initializer快速初始化一个Spring Boot工程
pom.xml
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
<exclusions>
<exclusion>
<groupId>org.junit.vintage</groupId>
<artifactId>junit-vintage-engine</artifactId>
</exclusion>
</exclusions>
</dependency>
<!--lombok用来简化实体类-->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
<!--mybatis-plus-->
<dependency>
<groupId>com.baomidou</groupId>
<artifactId>mybatis-plus-boot-starter</artifactId>
<version>3.0.5</version>
</dependency>
<!--mysql-->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.38</version>
</dependency>
</dependencies>
配置文件application.properties,配置数据库相关配置
#mysql数据库连接(mysql5.x版本)
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.url=jdbc:mysql://localhost:3306/mybatis_plus?useUnicode=true&characterEncoding=utf-8&serverTimezone=Asia/Shanghai
spring.datasource.username=root
spring.datasource.password=1234
#mybatisplus日志
mybatis-plus.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl
在SpringBoot启动类上添加@MapperScan主键,扫码mapper文件夹(也可以放在MybatisPlus配置文件)
@SpringBootApplication
@MapperScan("com.guli.mybatisplusdemo.mapper")
public class MybatisPlusDemoApplication {
public static void main(String[] args) {
SpringApplication.run(MybatisPlusDemoApplication.class,args);
}
}
创建实体User.java
package com.guli.mybatisplusdemo.entity;
import lombok.Data;
/**
* @author Guli
* @date 2020/4/2 13:52
*/
@Data
public class User {
private Long id;
private String name;
private Integer age;
private String email;
}
因为是demo,省掉controller和service,直接写mapper:UserMapper.java
public interface UserMapper extends BaseMapper<User> {
} //BaseMapper接口封装了很多方法可以使用
1)查询操作
@SpringBootTest
public class MpDemoTest {
@Autowired
private UserMapper userMapper;//此处如果报红可以不管,强迫症可以在UserMapper接口上加@Repository注解
//查询user表所有的数据
@Test
public void findAll(){
List<User> users = userMapper.selectList(null);
users.forEach(System.out::println);
}
}
运行查看结果...
2)插入操作
@Test
public void testInsert(){
User user = new User();
user.setName("张三");
user.setAge(11);
user.setEmail("zhangsan@qq.com");
int insert = userMapper.insert(user);
System.out.println(insert); //返回受影响的行数
}
运行结果...
mybatis-plus默认的主键策略是:ID_WORKER (全局唯一ID)
3) 更新操作
@Test
public void testUpdate()
User user = new User();
user.setId(1L);
user.setAge(100);
int i = userMapper.updateById(user);
System.out.println("i = " + i);
}
更新操作前:
更新操作之后:
一些数据每次都使用相同的方式填充,我们例如创建时间、更新时间等,都可以使用Mybatis-Plus的自动填充功能实现。
@Data
public class User {
@TableId(type= IdType.ID_WORKER) //mp默认的id策略,生成19位,Long类型id使用这个
// @TableId(type = IdType.ID_WORKER_STR) //mp自带的策略,生成19位值,如果id是string类型使用这个
private Long id;
private String name;
private Integer age;
private String email;
@TableField(fill = FieldFill.INSERT)
private Date createTime;
@TableField(fill = FieldFill.INSERT_UPDATE)
private Date updateTime;
}
3. 实现对元对象处理器接口java
@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);
}
}
然后执行更新之前的添加和更新操作就可以了..
乐观锁的适用场景:当更新操作某一条数据的时候,这条数据不被其他线程更新,实现线程安全的数据更新。
实现方式:
(1)在User表中添加version字段,数据类型int 长度11(设置默认值1)
(2)在实体类中添加version字段并添加@Version注解
@Version
@TableField(fill = FieldFill.INSERT)
private Integer version;
(3)在MybatisPlusConfig中注册Bean
注意这个地方可以把@MapperScan注解从启动类移到这里,这里有启动类上就可以删除
@EnableTransactionManagement
@Configuration
@MapperScan("com.guli.mybatisplusdemo.mapper")
public class MybatisPlusConfig {
/**
* 乐观锁插件
*/
@Bean
public OptimisticLockerInterceptor optimisticLockerInterceptor(){
return new OptimisticLockerInterceptor();
}
}
(4)测试乐观锁修改
@Test
public void testOptimisticLock(){
//第一步:查询出数据
User user = userMapper.selectById(1L);
//第二步:修改数据
user.setName("李四");
user.setEmail("lisi@qq.com");
//第三步执行更新操作
userMapper.updateById(user);
}
可以看的version已经由1变成了2
(1)在数据库中添加deleted字段,类型为tinyint,长度为1
(2)在User实体类中添加deleted字段
@TableLogic
@TableField(fill = FieldFill.INSERT)
private Integer deleted;
(3)元对象处理器接口中添加deleted的insert默认值(也可以在数据库设计表的时候默认值设为0)
@Override
public void insertFill(MetaObject metaObject) {
this.setFieldValByName("createTime",new Date(),metaObject);
this.setFieldValByName("updateTime",new Date(),metaObject);
this.setFieldValByName("deleted",0,metaObject);
}
(4)mybatis-plus默认0是没有删除,1是删除,也可以在配置文件中修改
mybatis-plus.global-config.db-config.logic-delete-value=1
mybatis-plus.global-config.db-config.logic-not-delete-value=0
(5)在MybatisPlusConfig中注册Bean
@Bean
public ISqlInjector sqlInjector(){
return new LogicSqlInjector();
}
(6)测试逻辑删除逻辑
@Test
public void testLogicDeleteSelect(){
int i = userMapper.deleteById(1L);
System.out.println("i = " + i);
}
执行成功可以看到数据量id为1那条数据deleted字段值变为1;
然后调用查询所有的方法查看结果:
可以看到mybatis-plus会自动的在sql语句中添加WHERE deleted=0,查询出的结果里也没有刚刚删除的数据。
Wrapper :条件构造抽象类,最顶端父类
AbstractWrapper :用于查询条件封装,生成 sql 的 where 条件
QueryWrapper :Entity 对象封装操作类,不是用lambda语法
UpdateWrapper :Update 条件封装,用于Entity对象更新操作
AbstractLambdaWrapper :Lambda 语法使用 Wrapper统一处理解析 lambda 获取 column。
LambdaQueryWrapper :看名称也能明白就是用于Lambda语法使用的查询Wrapper
LambdaUpdateWrapper :Lambda 更新封装Wrapper
(1)gt、ge、lt、le
//查询年龄大于等于20的user
@Test
public void testGe(){
List<User> users = userMapper.selectList(new QueryWrapper<User>().lambda().ge(User::getAge, 20));
users.forEach(System.out::println);
}
(2)eq、ne
//查询name是张三
@Test
public void testEq(){
User user = userMapper.selectOne(new QueryWrapper<User>().lambda().eq(User::getName, "张三"));
System.out.println("user = " + user);
}
(3)between区间
//查询年龄在20-30之间的
@Test
public void testBetween(){
List<User> users = userMapper.selectList(new QueryWrapper<User>().lambda().between(User::getAge, 20, 30));
users.forEach(System.out::println);
}
(4)like 模糊查询
//对名字进行模糊查询
@Test
public void testLike(){
List<User> user = userMapper.selectList(new QueryWrapper<User>().lambda().like(User::getName, "张"));
user.forEach(System.out::println);
}
还有很多很方便使用的功能可以上官网学习...
往期回顾: