Bootstrap

MyBatisPlus

概述

  • MyBatisPlus(简称MP)是基于MyBatis框架基础上开发的增强型工具,旨在简化开发,提高效率
  • 开发方式
    • 基于MyBatis使用MyBatisPlus
    • 基于Spring使用MyBatisPlus
    • 基于SpringBoot使用MyBatisPlus
  • 官网 :https://baomidou.com/

特性

  • 无侵入:只做增强不做改变,引入它不会对现有工程产生影响,如丝般顺滑
  • 强大的 CRUD 操作:内置通用 Mapper、通用 Service,仅仅通过少量配置即可实现单表大部分 CRUD 操作,更有强大的条件构造器,满足各类使用需求
  • 支持 Lambda 形式调用:通过 Lambda 表达式,方便的编写各类查询条件,无需再担心字段写错
  • 支持主键自动生成:支持多达 4 种主键策略(内含分布式唯一 ID 生成器 - Sequence),可自由配置,完美解决主键问题
  • 内置分页插件:基于 MyBatis 物理分页,开发者无需关心具体操作,配置好插件之后,写分页等同于普通 List 查询

入门案例

  • SpringBoot整合MyBatis开发过程(复习)
    • 创建SpringBoot工程
    • 勾选配置使用的技术
    • 设置DataSource的相关属性(JDBC参数)
    • 定义数据层接口映射配置

步骤:SpringBoot整合MyBatisPlus入门程序

①创建模块(省略)

②选择当前模块需要使用的技术集合(仅保留JDBC,不勾选MyBatisFramework)

③手动添加起步依赖

<!--      TODO  mybatis-plus-->
<dependency>
    <groupId>com.baomidou</groupId>
    <artifactId>mybatis-plus-boot-starter</artifactId>
    <version>3.5.2</version>
</dependency>
<!--        TODO druid-->
<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>druid</artifactId>
    <version>1.2.11</version>
</dependency>
  • 由于mp并未被收录到idea的系统内置配置,无法直接选择加入

④设置Jdbc参数(application.yml)

spring:
  datasource:
    driver-class-name: com.mysql.cj.jdbc.Driver
    url: jdbc:mysql://localhost:3306/test
    username: root
    password: 12345678
    type: com.alibaba.druid.pool.DruidDataSource
  • 注意:如果使用Druid数据源,则需要导入对应坐标

⑤制作实体类与表结构(类名与表名对应,属性名与字段名对应)

  • 注意这里的主键必须为bigint,因为springboot的底层会做转换
/*
 Navicat Premium Data Transfer

 Source Server         : sa
 Source Server Type    : MySQL
 Source Server Version : 80028
 Source Host           : localhost:3306
 Source Schema         : test

 Target Server Type    : MySQL
 Target Server Version : 80028
 File Encoding         : 65001

 Date: 29/07/2022 01:37:51
*/

SET NAMES utf8mb4;
SET FOREIGN_KEY_CHECKS = 0;

-- ----------------------------
-- Table structure for user
-- ----------------------------
DROP TABLE IF EXISTS `user`;
CREATE TABLE `user`  (
  `id` bigint NOT NULL AUTO_INCREMENT,
  `name` char(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `password` char(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `age` int NULL DEFAULT NULL,
  `tel` char(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 1 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Dynamic;

-- ----------------------------
-- Records of user
-- ----------------------------
INSERT INTO `user` VALUES (1, 'null', 'admin', 1, '123456');
INSERT INTO `user` VALUES (2, 'heimachengxuyuan', '12345678', 18, '123456789');
INSERT INTO `user` VALUES (1552709826475171842, NULL, NULL, NULL, NULL);

SET FOREIGN_KEY_CHECKS = 1;

public class User {
    private Long id;
    private String name;
    private String password;
    private Integer age;
    private String tel;
}

⑥定义数据接口,继承BaseMapper<User>

@Mapper
public interface UserDao extends BaseMapper<User> {

}

⑦在测试类中注入dao接口,测试功能

@SpringBootTest
class Mybatis01QuickstartApplicationTests {
    @Autowired
    private UserDao userDao;

    @Test
    void testGetAll() {
        List<User> list = userDao.selectList(null);
        System.out.println(list);
    }

}

标准CRUD功能

在这里插入图片描述

lombok

  • Lombok,一个Java类库,提供了一组注解,简化POJO实体类开发
<!--        TODO lombok-->
<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <version>1.18.24</version>
    <scope>provided</scope>
</dependency>
  • 常用注解:@Data

  • //lomok
    //@Setter
    //@Getter
    //@ToString
    //@NoArgsConstructor
    //@AllArgsConstructor
    //@EqualsAndHashCode
    @Data
    public class User {
        private Long id;
        private String name;
        private String password;
        private Integer age;
        private String tel;
    }
    
  • 为当前实体类在编译期设置对应的get/set方法,无参/有参构造方法,toString()方法,hashCode方法,equals方法等

MP分页查询功能

①设置分页拦截器作为Spring管理的bean

@Configuration
public class MpConfig {
    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {
        //1.定义Mp拦截器
        MybatisPlusInterceptor mybatisPlusInterceptor = new MybatisPlusInterceptor();
        //2.添加具体的拦截器
        mybatisPlusInterceptor.addInnerInterceptor(new PaginationInnerInterceptor());
        return mybatisPlusInterceptor;
    }
}

②执行分页查询

@Test
void testGetByPage() {
    IPage page = new Page(1, 5);
    userDao.selectPage(page, null);
    System.out.println("当前页码值:" + page.getCurrent());
    System.out.println("当前显示数:" + page.getSize());
    System.out.println("一共多少页" + page.getPages());
    System.out.println("一共多少条数据" + page.getTotal());
    System.out.println("数据" + page.getRecords());
}
  • 开启日志(可以不开启)
mybatis-plus:
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

DQL编程控制

条件查询

  • MyBatisPlus将书写复杂的SQL查询条件进行了封装,使用编程的形式完成查询条件的组合

条件查询----设置查询条件

    @Test
    void testGetAll() {
//        方式一:按条件查询
//        QueryWrapper qw = new QueryWrapper();
//        qw.lt("age", 18);
//        List<User> userList = userDao.selectList(qw);
//        System.out.println(userList);

//        //方式二:Lambda格式按条件查询
//        QueryWrapper<User> qw = new QueryWrapper<User>();
//        qw.lambda().lt(User::getAge,20);
//        List<User> userList = userDao.selectList(qw);
//        System.out.println(userList);

        //方式三:Lambda格式按条件查询
//        LambdaQueryWrapper<User> lqw = new LambdaQueryWrapper<User>();
//        lqw.lt(User::getAge,20);
//        List<User> userList = userDao.selectList(lqw);
//        System.out.println(userList);

        LambdaQueryWrapper<User> lqw = new LambdaQueryWrapper<User>();
//        lqw.lt(User::getAge,20);
//        lqw.gt(User::getAge,10);
        //链式编程--and
//        lqw.lt(User::getAge, 20).gt(User::getAge, 10);
        //或者or
        lqw.lt(User::getAge, 10).or().gt(User::getAge, 15);
        List<User> userList = userDao.selectList(lqw);
        System.out.println(userList);
    }

条件查询–null值处理

//模拟页面传递过来的查询数据
UserQuery uq = new UserQuery();
uq.setAge(10);
uq.setAge2(30);

//null判定
//1.使用if判断
//2.参数:添加一个条件
LambdaQueryWrapper<User> lqw = new LambdaQueryWrapper<User>();
//先判定第一个参数是否为true,如果为true连接当前条件
lqw.lt(null != uq.getAge2(), User::getAge, uq.getAge2())
        .gt(null != uq.getAge(), User::getAge, uq.getAge());
List<User> userList = userDao.selectList(lqw);
System.out.println(userList);

查询投影

        //查询投影
//        LambdaQueryWrapper<User> lqw = new LambdaQueryWrapper<User>();
//        //设置只查哪个字段,并且只适用于匿名表达式
//        lqw.select(User::getId, User::getName, User::getAge);
        //非匿名表达式写法
//        QueryWrapper<User> qw = new QueryWrapper<User>();
//        qw.select("id","name","age","tel");
//        List<User> userList = userDao.selectList(qw);
//        System.out.println(userList);


        QueryWrapper<User> qw = new QueryWrapper<User>();
        qw.select("count(*) as count,tel");
        qw.groupBy("tel");
        List<Map<String, Object>> maps = userDao.selectMaps(qw);
        System.out.println(maps);
  • 查询结果包含模型类中部分属性
LambdaQueryWrapper<User> lqw = new LambdaQueryWrapper<User>();
lqw.select(User::getId,User::getName,User::getAge);
List<User> userList = userDao.selectList(lqw);
System.out.println(userList);
  • 查询结果包含模型类中未定义的属性
QueryWrapper<User> qm = new QueryWrapper<User>();
qm.select("count(*) as nums,gender");
qm.groupBy("gender");
List<Map<String,Object>> maps = userDao.selectMaps(qm);
System.out.println(maps);

查询条件

        //条件查询
//        LambdaQueryWrapper<User> lqw = new LambdaQueryWrapper<User>();
//        //等同于=
//        lqw.eq(User::getName,"jerry").eq(User::getPassword,"jerry");
//        User user = userDao.selectOne(lqw);
//        System.out.println(user);

//        //范围查询lt le gt ge eq between
//        LambdaQueryWrapper<User> lqw = new LambdaQueryWrapper<User>();
//        //前面必须是小的
//        lqw.between(User::getAge,10,20);
//        List<User> userList = userDao.selectList(lqw);
//        System.out.println(userList);

//        模糊匹配like likeRight--J% likeLeft--%J
        LambdaQueryWrapper<User> lqw = new LambdaQueryWrapper<User>();
        lqw.likeLeft(User::getName,"J");
        List<User> userList = userDao.selectList(lqw);
        System.out.println(userList);
  • 用户登录(eq匹配)
LambdaQueryWrapper<User> lqw = new LambdaQueryWrapper<User>();
lqw.eq(User::getName,userQuery.getAge()).eq(User::getPassword,userQuery.getPassword());
User loginUser = userDao.selectOne(lqw);
System.out.println(loginUser);
  • 购物设定价格区间、户籍设定年龄区间(le ge 匹配或between匹配)
LambdaQueryWrapper<User> lqw = new LambdaQueryWrapper<User>();
//方案一:设定上限下限
lwq.le(User::getAge,userQueery.getAge()).ge(User::getAge,userQuery.getAge2());
//方案二:设定范围
lqw.between(User::getAge,userQuery.getAge(),userQuery.getAge2());
List<User> userList = userDao.selectList(lqw);
System.out.println(userList);
  • 查信息,搜索新闻(非全文检索版:like匹配)
LambdaQueryWrapper<User> lqw = new LambdaQueryWrapper<User>();
lqw.likeLeft(User::geteTel,userQuery.getTel());
List<User> userList = userDao.selectList(lqw);
System.out.println(userList);
  • 统计报表(分组查询聚合函数)
QueryWrapper<User> qm = new QueryWrapper<User>();
qm.select("count(*) as nums,gender");
qm.groupBy("gender");
List<Map<String,Object>> maps = userDao.selectMaps(qm);
System.out.println(maps);

官网(条件构造器)API:

https://baomidou.com/pages/10c804/#abstractwrapper

字段映射与表名映射

  • 名称:@TableFiled
  • 类型:属性注解
  • 位置:模型类属性定义上方
  • 作用:设置当前属性对应的数据库表中的字段关系
  • 范例:
public class User{
    @TableField(value="pwd")
    private Stirng password;
}
  • 相关属性
    • value(默认):设置数据表字段名称
    • exist:设置属性在数据库表字段中是否存在,默认为true。此属性无法与value合并使用
    • select:设置属性是否参与查询,此属性与select()映射配置不冲突(select=false)

  • 名称:@TableName
  • 类型:类注解
  • 位置:模型类定义上方
  • 作用:设置当前类对应的数据库表的关系
  • 范例:
@tableName("tbl_user")
public class User{
    private Long id;
}
  • 相关属性
    • value(默认):设置数据库表名称

DML编程控制

id生成策略控制

  • 不同的表应用不同的id生成策略
    • 日志:自增(1,2,3,4,…)
    • 购物订单:特殊规则(FQ23948AK3843)
    • 外卖单:关联地区日志日期等信息( 10 04 20200314 34 91)
    • 关系表:可省略id

  • 名称:@TableId
  • 类型:属性注解
  • 位置:模型类中用于表示主键的属性定义上方
  • 作用:设置当前类中主键属性的生成策略
  • 范例:
public class User{
    @TableId(type=IdType.AUTO)
    private Long id;
}
  • 相关属性:
    • value:设置数据库主键名称
    • type:设置主键属性的生成策略,值参照IdType枚举类型

id生成策略控制

  • AUTO(0):使用数据库id自增策略控制id生成

  • NONE(1):不设置id自动生成策略

  • INPUT(2):用户手工输入id

  • ASSIGN_ID(3):雪花算法生成id(可兼容数据型与字符串型)(IDEA默认)

  • ASSIGN_UUID(4):以UUID生成算法作为id生成策略

  • 雪花算法:(64位整数)

    • 第一位占位符必定是0(如果是1表示为负数)
    • 时间戳(41位)精确到毫秒,但是算法与我们的时间戳不一样
    • 机器码(5+5):群组加机器
    • 序列号(12位)

表名前缀全局配置

mybatis-plus:
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
  global-config:
    banner: false
    db-config:
      id-type: assign_id
      table-prefix: tbl_

多记录操作

    @Test
    void testDelete() {
        List<Long> list = new ArrayList<>();
        list.add(1552924879866527746L);
        list.add(1552927500232777730L);
        userDao.deleteBatchIds(list);

        //同理
//        userDao.selectBatchIds();

    }
  • 根据主键删除多条记录

  • 根据主键查询多条记录

逻辑删除

  • 删除操作业务问题业务数据从数据库中丢弃
  • 逻辑删除:为数据设置是否为可用状态字段,删除时设置状态字段为不可用状态,数据保留在数据库中

步骤:

①数据库表中添加逻辑删除标记字段

在这里插入图片描述

②实体类中添加对应的字段,并设定当前字段为逻辑删除标记字段

    //逻辑删除字段,标记当前记录是否被删除
//    @TableLogic(value = "0", delval = "1")
    private Integer deleted;

③配置逻辑删除字面值

mybatis-plus:
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
  global-config:
    banner: false
    db-config:
      id-type: assign_id
      table-prefix: tbl_
      logic-delete-field: deleted
      logic-delete-value: 1
      logic-not-delete-value: 0

乐观锁

步骤:

①数据库表中添加锁标记字段

在这里插入图片描述

②实体类中添加对应字段,并设定当前字段为逻辑删除标记字段

@Version
private Integer version;

③配置乐观锁拦截器实现锁机制对应的动态SQL语句拼装

@Bean
public MybatisPlusInterceptor mybatisPlusInterceptor() {
    //1.定义Mp拦截器
    MybatisPlusInterceptor mybatisPlusInterceptor = new MybatisPlusInterceptor();
    //2.添加具体的拦截器
    mybatisPlusInterceptor.addInnerInterceptor(new PaginationInnerInterceptor());
    //3.添加乐观锁的拦截器
    mybatisPlusInterceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor());
    return mybatisPlusInterceptor;
}

④使用乐观锁机制在修改前必须先获取到对应数据的version方可以正常进行

//1.先通过要修改的数据id将当前数据查询出来
User user = userDao.selectById(2L);//version=3
User user2 = userDao.selectById(2L);//version = 3

//2.将要修改的属性逐一设置进去
user.setName("Jackaaa");
userDao.updateById(user);

user.setName("Jack999");
userDao.updateById(user2);

代码生成器(快速开发)

  • 模板:MyBatisPlus提供
  • 数据库相关配置:读取数据库获取信息
  • 开发者自定义配置:手工配置
package com.itheima;

import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.generator.AutoGenerator;
import com.baomidou.mybatisplus.generator.config.DataSourceConfig;
import com.baomidou.mybatisplus.generator.config.GlobalConfig;
import com.baomidou.mybatisplus.generator.config.PackageConfig;
import com.baomidou.mybatisplus.generator.config.StrategyConfig;

public class CodeGenerator {
    public static void main(String[] args) {
        //1.获取代码生成器的对象
        AutoGenerator autoGenerator = new AutoGenerator();

        //设置数据库相关配置
        DataSourceConfig dataSource = new DataSourceConfig();
        dataSource.setDriverName("com.mysql.cj.jdbc.Driver");
        dataSource.setUrl("jdbc:mysql://localhost:3306/test");
        dataSource.setUsername("root");
        dataSource.setPassword("12345678");
        autoGenerator.setDataSource(dataSource);

        //设置全局配置
        GlobalConfig globalConfig = new GlobalConfig();
        globalConfig.setOutputDir(System.getProperty("user.dir")+"/mybatisplus_04_generator/src/main/java");    //设置代码生成位置
        globalConfig.setOpen(false);    //设置生成完毕后是否打开生成代码所在的目录
        globalConfig.setAuthor("黑马程序员");    //设置作者
        globalConfig.setFileOverride(true);     //设置是否覆盖原始生成的文件
        globalConfig.setMapperName("%sDao");    //设置数据层接口名,%s为占位符,指代模块名称
        globalConfig.setIdType(IdType.ASSIGN_ID);   //设置Id生成策略
        autoGenerator.setGlobalConfig(globalConfig);

        //设置包名相关配置
        PackageConfig packageInfo = new PackageConfig();
        packageInfo.setParent("com.aaa");   //设置生成的包名,与代码所在位置不冲突,二者叠加组成完整路径
        packageInfo.setEntity("domain");    //设置实体类包名
        packageInfo.setMapper("dao");   //设置数据层包名
        autoGenerator.setPackageInfo(packageInfo);

        //策略设置
        StrategyConfig strategyConfig = new StrategyConfig();
        strategyConfig.setInclude("tbl_user");  //设置当前参与生成的表名,参数为可变参数
        strategyConfig.setTablePrefix("tbl_");  //设置数据库表的前缀名称,模块名 = 数据库表名 - 前缀名  例如: User = tbl_user - tbl_
        strategyConfig.setRestControllerStyle(true);    //设置是否启用Rest风格
        strategyConfig.setVersionFieldName("version");  //设置乐观锁字段名
        strategyConfig.setLogicDeleteFieldName("deleted");  //设置逻辑删除字段名
        strategyConfig.setEntityLombokModel(true);  //设置是否启用lombok
        autoGenerator.setStrategy(strategyConfig);
        //2.执行生成操作
        autoGenerator.execute();
    }
}
package com.southwind;

import com.baomidou.mybatisplus.annotation.DbType;
import com.baomidou.mybatisplus.generator.AutoGenerator;
import com.baomidou.mybatisplus.generator.config.DataSourceConfig;
import com.baomidou.mybatisplus.generator.config.GlobalConfig;
import com.baomidou.mybatisplus.generator.config.PackageConfig;
import com.baomidou.mybatisplus.generator.config.StrategyConfig;
import com.baomidou.mybatisplus.generator.config.rules.NamingStrategy;

public class GenerateTest {
    public static void main(String[] args) {
        //创建generator对象
        AutoGenerator autoGenerator = new AutoGenerator();
        //数据源
        DataSourceConfig dataSourceConfig = new DataSourceConfig();
        dataSourceConfig.setDbType(DbType.MYSQL);
        dataSourceConfig.setDriverName("com.mysql.cj.jdbc.Driver");
        dataSourceConfig.setUsername("root");
        dataSourceConfig.setPassword("123456");
        dataSourceConfig.setUrl("jdbc:mysql://localhost:3306/test11");
        autoGenerator.setDataSource(dataSourceConfig);
        //全局配置
        GlobalConfig globalConfig = new GlobalConfig();
        globalConfig.setOutputDir(System.getProperty("user.dir")+"/src/main/java");
        globalConfig.setAuthor("admin");
        globalConfig.setOpen(false);
        globalConfig.setServiceName("%sService");
        autoGenerator.setGlobalConfig(globalConfig);
        //包信息
        PackageConfig packageConfig = new PackageConfig();
        packageConfig.setParent("com.southwind");
        packageConfig.setEntity("entity");
        packageConfig.setMapper("mapper");
        packageConfig.setService("service");
        packageConfig.setServiceImpl("service.impl");
        packageConfig.setController("controller");
        autoGenerator.setPackageInfo(packageConfig);
        //策略配置
        StrategyConfig strategyConfig = new StrategyConfig();
        strategyConfig.setInclude("fruit");
        strategyConfig.setNaming(NamingStrategy.underline_to_camel);
        strategyConfig.setColumnNaming(NamingStrategy.underline_to_camel);
        strategyConfig.setEntityLombokModel(true);
        autoGenerator.setStrategy(strategyConfig);
        //运行
        autoGenerator.execute();
    }
}



<!--        代码生成器,注意版本号,3.5.1之后的版本不能使用这一套模板,详情见官方开发-->
<dependency>
    <groupId>com.baomidou</groupId>
    <artifactId>mybatis-plus-generator</artifactId>
    <version>3.4.1</version>
</dependency>

<!--        velocity模板引擎-->
<dependency>
    <groupId>org.apache.velocity</groupId>
    <artifactId>velocity-engine-core</artifactId>
    <version>2.3</version>
</dependency>
;