MyBatisPlus(MP)学习记录(分页查询的开启+日志打印配置+乐观锁+映射匹配+字段匹配和表名匹配+null值判定+代码生成器等等)

news2025/1/18 8:49:50

MP介绍(官网链接)

MyBatis-Plus(简称 MP)是一个 MyBatis 的增强工具,在 MyBatis 的基础上只做增强不做改变,为简化开发、提高效率而生。

前言

本篇文章展示的MP教学中涉及到的版本如下:

  • IDEA :2022.1
  • SpringBoot : 2.7.7
  • Java : 1.8
  • DataSource :com.alibaba:druid 1.1.16
  • MyBatisPlus :3.4.1
  • MySQL :8.0.26

入门

1.创建SpringBoot项目
参考博主的IDEA创建SpringBoot项目教程这篇博文,需要注意一点,选择boot版本的时候尽量选择3.0以下的,否则低版本的java将创建失败,另外一点,提供的依赖只需选择MySQL Driver即可,因为本篇文章只是MP的入门,一个MySQL的依赖就足够的。
依赖说明
2.相关配置

mybatis-plus依赖
<dependency>
	<groupId>com.baomidou</groupId>
	<artifactId>mybatis-plus-boot-starter</artifactId>
	<version>3.4.1</version>
</dependency>

数据源Druid依赖
<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>druid</artifactId>
    <version>1.1.16</version>
</dependency>

数据库配置信息:
数据库配置信息
3. 单独配置分页功能
配置类的书写:

package org.zzzzzz.config;

import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class MpConfig {
    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() { // 分页拦截器的添加
        MybatisPlusInterceptor mpInterceptor = new MybatisPlusInterceptor();
        mpInterceptor.addInnerInterceptor(new PaginationInnerInterceptor());
        return mpInterceptor;
    }
}

4.在测试类中进行分页方法测试

@Test
    public void testGetByPage() {
        // 第二页,每页四条数据
        IPage page = new Page(2, 4);
        bookDao.selectPage(page, null);
        page.getPages(); // 一共有多少页
        page.getRecords(); // 查询出的数据
        page.getTotal(); // 所有的数据
        page.getCurrent(); // 当前页数
        page.getSize(); // 每页的大小
    }

日志的开启(方便调试):

mybatis-plus:
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

查看测试信息(包含日志)
测试分页的结果
5.条件查询

  • 条件查询格式
        // 方式一
        QueryWrapper qw = new QueryWrapper();
        qw.lt("id", 7);
        List<Book> list = bookDao.selectList(qw);
       	System.out.println(list);

		// 方式二
        QueryWrapper<Book> qw = new QueryWrapper();
        qw.lambda().lt(Book::getId, 7);
        List<Book> list = bookDao.selectList(qw);
        System.out.println(list);

        // 方式三 :链式编程
        LambdaQueryWrapper<Book> lqw = new LambdaQueryWrapper<Book>();
		// lqw.lt(Book::getId, 5).gt(Book::getId, 7);  并且关系
        lqw.gt(Book::getId, 10).or().lt(Book::getId, 5); // 或者关系
        List<Book> list = bookDao.selectList(lqw);
        System.out.println(list);
  • 条件查询null值判定方式
		BookQuery bookQuery = new BookQuery();
		bookQuery.setId(xx);  // Id上限 
		bookQuery.setId2(yy); // Id下限
		LambdaQueryWrapper<Book> lqw = new LambdaQueryWrapper<Book>();
		// 设置下限
		lqw.gt(null != bookQuery.getId2(), Book::getId, bookQuery.getId2());
		// 设置上限
        lqw.lt(null != bookQuery.getId(),  Book::getId, bookQuery.getId());
        
        List<Book> list = bookDao.selectList(lqw);
        System.out.println(list);
  • 查询投影
    查询结果包含模型类中部分属性
		// 使用Lambda的方式
		LambdaQueryWrapper<Book> lqw = new LambdaQueryWrapper<Book>();
        lqw.select(Book::getId, Book::getName); // 查询指定字段
        List<Book> list = bookDao.selectList(lqw);
        System.out.println(list);

        查询结果包含模型类中未定义的属性

		// 只能使用常规方式
		QueryWrapper<Book> qw = new QueryWrapper();
        qw.select("count(*) as count, type"); // 查询数据的条数
        qw.groupBy("type");
        List<Map<String, Object>> mapList = bookDao.selectMaps(qw);
        System.out.println(mapList);

        其他复杂情况回归到最开始在Dao接口中写SQL的时候(MyBatis)

		// 无代码,需要依据具体情况写SQL语句
  • 查询条件
    条件过多,查询官方文档即可。
    https://baomidou.com/pages/10c804/#abstractwrapper
  • 映射匹配
    解决的问题:数据库表的字段名与pojo类中的属性名不一致、表名与pojo类名不对称导致查询时不匹配、某字段不参与查询(考虑安全性:密码、身份证号、手机号等等)、属性值后续添加但不在表字段中等等。
    映射匹配
  • id生成策略(MP中的枚举类)
    注解格式:@TableId(type= IdType.AUTO)
    idType
    注:也可以设置xml文件里的MP全局配置(大量实体类的规范(比较广,意会一下即可)一致的情况下)
  • 批量删除(id)与批量查询(id)
    没有什么特别要指出来的,就是两个api的调用。
  • 逻辑删除(给每条数据增加一个标记字段(deleted),以及实体类增加字段)
    需要设定被删除的记录,他们的标记是什么(一般是1),没被删除的记录,他们的标记是什么(一般是0)。这些只需要在xml配置中写即可。
    逻辑删除配置信息
    注:一旦进行删除操作,那么实际的SQL语句为UPDATA语句,而不是DELETE语句,它会使用UPDATA语句进行标记的更新,实现逻辑删除。
    注:如果想要查询那些已经被逻辑删除的记录,需要自己手写Dao层的SQL语句。
  • MP添加乐观锁(解决小型并发问题:秒杀、抢票等等)
    表中与实体类中添加字段version(也算是一个标记),在MP的配置类中添加一个乐观锁的拦截器。(注:要想使乐观锁启用,对记录进行修改的时候,一定要包含字段version。在开发中一般是先查询后更新,遵循了上面的规则。)
  • MP代码生成器(模板+自定义配置)
    依赖的导入:
		<!--代码生成器-->
        <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>

代码自动生成配置CodeGenerator.java

package org.zzzzzz;

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/ssm_db?serverTimezone=UTC");
        dataSource.setUsername("root");
        dataSource.setPassword("root");
        autoGenerator.setDataSource(dataSource);

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

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

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/158578.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Unity 之 Addressable可寻址系统 -- 可寻址系统概念介绍 -- 入门(一)

可寻址系统面板概念 -- 入门&#xff08;一&#xff09;一&#xff0c;可寻址系统概念介绍1.1 官方话术1.2 几个概念二&#xff0c;可寻址系统目录介绍2.1 导入工程2.2 目录介绍概述&#xff1a;本片文章带大家了解可寻址系统的相关概念&#xff0c;为大家介绍可寻址系统导入方…

Elasticsearch:如何在 Docker 上运行 Elasticsearch 8.x 进行本地开发

对于 Elasticsearch 的新主要版本 (8.x.x)&#xff0c;有关于在 Docker 上运行 Elasticsearch 和 Kibana 的重大更新。 过去适用于以前版本的 Docker 和 Docker Compose 的命令和语法需要更新才能适用于最新版本。 在这篇文章中&#xff0c;我们将介绍如何使用 Docker 和 Docke…

Hudi系列5:Hudi与Spark集成

文章目录一. Spark操作Huid概述二. Spark-Shell方式三. pyspark方式四. SparkSQL方式参考:一. Spark操作Huid概述 Hudi各个版本支持Spark版本情况: 将hudi集成spark的jar包&#xff0c;放到spark的jars目录 -- 同步到其它节点 cd /home/hudi-0.12.0/packaging/hudi-spark-b…

【设计模式】UML类图

设计模式学习之旅(一) 查看更多可关注后查看主页设计模式DayToDay专栏 一.UML概述 统一建模语言&#xff08;Unified Modeling Language&#xff0c;UML&#xff09;&#xff0c;是一种用于软件系统分析和设计的语言工具。特点是简单、统一、图形化、能表达软件设计中的动态与静…

Matlab论文插图绘制模板第72期—金字塔图

前段时间&#xff0c;有朋友觉得一篇文章里的插图看着不错&#xff0c;问Matlab能不能实现&#xff1a; 想了想&#xff0c;Matlab中并没有现成的画图代码。 但本质上&#xff0c;这种图其实就是横向柱状图的变体&#xff0c;而横向柱状图的绘制模板之前有分享过&#xff1a; …

Java多线程:线程的优先级 || 线程的中断 || 线程的分类 || 线程的定时器

Java多线程&#xff1a;线程的优先级 || 线程的中断 || 线程的分类 || 线程的定时器 每博一文案 正如村上春树所说&#xff1a;“当暴风雨过去&#xff0c;你不会记得自己是如何度过的&#xff0c;你甚至不确定&#xff0c;暴风雨是否真正结束了&#xff0c; 但你已不再是当初…

分布式唯一ID的几种生成方案,一次性全掌握!

上一篇文章&#xff0c;我们聊了一下分库分表相关的一些基础知识&#xff0c;具体可以参见&#xff1a;《用真实业务场景告诉你&#xff0c;高并发下如何设计数据库架构&#xff1f;》。 这篇文章&#xff0c;我们就接着分库分表的知识&#xff0c;来具体聊一下全局唯一id如何…

Flutter 基础-上

一、初始化项目 Material Design (Google 推出的前端UI 解决方案) 官网: https://www.material.io/中文网: https://material-io.cn/ Flutter 中一切内容都是组件(Widget) 无状态组件(StatelessWidget)有状态组件(StatefulWidget) 二、app结构 MaterialApp title (任务管理器…

【VScode技巧】:platformio部署ESP32Cam开发板

学习目标&#xff1a; 前几天用Arduino开发了ESP32Cam&#xff0c;实现了人脸识别的功能&#xff0c;今天无聊想了想ESP32Can也可以在VScode的Platformio中使用&#xff0c;于是就试着配置了一下。以下是配置环境的过程&#xff0c;谢谢大家观看。 正所谓工欲善其事&#xff0c…

【零基础】学python数据结构与算法笔记10

文章目录前言57.链表介绍58.链表的创建和遍历59.链表的插入和删除60.双链表61.链表总结62.哈希表62.哈希表实现64.哈希表应用总结前言 学习python数据结构与算法&#xff0c;学习常用的算法&#xff0c; b站学习链接 57.链表介绍 链表是由一系列节点组成的元素集合。每个节点…

React配置文件(五) 配置less

代码如下: module.exports { webpack: override( addLessLoader({ lessOptions: { javascriptEnabled: true, modifyVars: { primary-color: #1DA57A }, }, }), adjustStyleLoaders(({ use: [, , postcss] }) > { const postcssOptions postcss.options postcss.options …

SEO初学者如何快速做好 SEO 优化?seo数据查询

昨天给大家介绍了seo的意义和重要性&#xff0c;今天让我们一起看看10个基本的SEO初学者技巧&#xff0c;如何优化网站以增加流量。 1. 研究关键词并使用尾词 关键词在SEO中起着重要的作用。关键字表明了你文章的主要主题&#xff0c;它使人们有可能在网上搜索感兴趣的主题时找…

RK3588平台开发系列讲解(日志篇)RK3588 syslog的使用

平台内核版本安卓版本RK3588Linux 5.10Android 12文章目录 一、查看是否启用syslog.conf二、配置启用syslog.conf1、配置busybox2、添加配置文件3、编译buildroot烧录三、验证1、编写测试代码2、查看日志文件3、运行测试程序沉淀、分享、成长,让自己和他人都能有所收获!😄 …

SpringCloudAlibaba入门(2023版)

先知 架构图一览 创建Serve端 新建项目 配置文件 application.yaml server:port: 8080# Eureka配置 eureka:instance:## Eureka实例的名称hostname: localhostAclient:# false表示自己端就是注册中心&#xff0c;职责就是维护服务实例&#xff0c;并不需要去检查服务fetch-r…

Kafka-eagle 安装教程

参考资料&#xff1a; 参考视频 Kafka-eagle官网 解释&#xff1a;Kafka-eagle 后来更名为 EFAK &#xff0c;所以打开官网会显示EFAK 环境准备&#xff1a; 要有本服务器可访问的MySQL服务&#xff0c;远程的请检查是否已经开启了MySQL远程访问&#xff0c;如果没有则需…

1.12 PWM实验

蜂鸣器--------TIM4 CH1/PB6 风扇-----------TIM1 CH1/PE9 马达-----------TIM16 CH1/PF6 一.PWM框图 RCC&#xff1a;使能GPIOB GPIOB&#xff1a;设置复用功能 TIM4:产生方波 二.分析RCC 确定总线&#xff0c;使能GPIOB和TIM4&#xff0c;GPIOE和TIM1,GPIO和TIM16 三.分…

【C++11】右值引用和移动语义

目录 一、左值 vs 右值 二、左值引用vs 右值引用 三、右值引用使用场景和意义 1. 左值引用的使用场景 2. 左值引用的短板 3. 右值引用和移动语义解决上述问题 四、右值引用引用左值的使用场景 五、完美转发 1. 模板中的&& 万能引用 2. std::forward 完美转发…

windows排查问题常用命令

查看JAVA进程占用PID: wmic process where caption"java.exe" get processid,caption,commandline /value查看进程端口信息&#xff1a;netstat -ano 或者 netstat -ano|findstr "8080" 或查看成功建立连接的&#xff1a;netstat -ano | findstr “ESTABLI…

C语言中的字符指针

目录1.字符指针指向一个字符2.字符指针指向一个字符串3.例题1.字符指针指向一个字符 int main() {char ch w;char *pc &ch;return 0; }将一个char类型的变量的地址放到一个char*类型的指针里去&#xff0c;这里的char*ps就是字符指针 在这里的字符指针与之前的整形指针等…

巧用gitbash的scp命令实现跨网段的文件直传

背景 嵌入式开发的工作流一般是这样的&#xff0c;程序员通过Windows电脑登陆Linux服务器&#xff0c;在服务器上编译出二进制文件后&#xff0c;先将文件scp到本地&#xff0c;然后再scp到Linux开发板&#xff0c;如下图所示 这样做需要执行两次scp命令&#xff0c;能否只执…