MyBatis-Plus速成指南:条件构造器和常用接口

news2025/2/5 18:34:15


Wrapper 介绍

  1. Wrapper:条件构造抽象类,最顶端父类
    1. AbstractWrapper:用于查询条件封装,生成 SQL 的 where 条件
    2. QueryWrapper:查询条件封装
    3. UpdateWrapper:Update 条件封装
    4. AbstractLambdaWrapper:使用 Lambda 语法
      1. LambdaQueryWrapper:使用 Lambda 语法
      2. LambdaUpdateWrapper:Lambda 更新封装 Wrapper

QueryWrapper:

  1. 组装条件查询:
    //组装查询条件
    @Test
    public void test(){
        //查询用户包含 a,年龄在20到30之间,并且邮箱不为 null 的用户信息
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        queryWrapper.like("name","a")
                .between("age",20,30)
                .isNotNull("email");
        userMapper.selectList(queryWrapper).forEach(System.out::println);
    }
  2. 组装排序条件:
    //组装排序条件
    @Test
    public void test02(){
        //按年龄降序查询用户,如果年龄相同则按 id 升序排序
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        queryWrapper.orderByDesc("age")
                .orderByAsc("id");
        userMapper.selectList(queryWrapper).forEach(System.out::println);
    }
  3. 组装条件删除:
    //组装删除条件
    @Test
    public void void03(){
        //删除 email 为空的数据
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        queryWrapper.isNull("email");
        int result = userMapper.delete(queryWrapper);
        System.out.println(result);
    }
  4. 条件的优先级:
    //条件的优先级
    @Test
    public void test04(){
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        //将 (年龄大于 20并且用户名中包含 a) 或邮箱为 null的用户信息修改
        queryWrapper.like("name","a")
                .gt("age",20)
                .or()
                .isNull("email");
        User user = new User();
        user.setAge(18);
        user.setEmail("1@163.com");
        int result = userMapper.update(user,queryWrapper);
        System.out.println(result);
    }
  5. 组装 select 语句:
    //组装 select 子句
    @Test
    public void test05(){
        //查询用户信息的 name 和 age 字段
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        queryWrapper.select("name","age");
        //selectMaps() 返回 Map 集合列表,通常配合 select() 使用,避免 User 对象中
        //没有被查询到的列值为 null
        userMapper.selectMaps(queryWrapper).forEach(System.out::println);
    }
  6. 实现子查询:
    @Test
    public void test06(){
        //查询 id 小于等于 3 的用户信息
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        queryWrapper.inSql("id","select id from t_user where id <= 3");
        userMapper.selectList(queryWrapper).forEach(System.out::println);
    }

UpdateWrapper:

//UpdateWrapper
@Test
public void test07(){
    //将(年龄大于 20 或邮箱为 null) 并且用户名中包含有 a 的用户信息修改
    //组装 set 子句以及修改条件
    UpdateWrapper<User> updateWrapper = new UpdateWrapper<>();
    //lambda 表达式内的逻辑优先运算
    updateWrapper.set("age",18)
            .set("email","user@163.com")
            .like("name","a")
            .and(i -> i.gt("age", 20).or().isNull("email"));
    //注意:
        //这里必须创建 User 对象,否则无法应用自动填充. 如果没有自动填充,可以设置为null,
        //User user = new User();
        //user.setName("张三");

    int result = userMapper.update(null,updateWrapper);
    System.out.println(result);
}

Condition:

  1. 概述:
    1. 在真正开发过程中,组装条件是常见的功能,而这些条件来源于用户输入,是可选的
    2. 因此我们在组装这些条件时,必须先判断用户是否选择了这些条件,若选择则需要组装这些条件,若没有选择一定不能组装,以免影响 SQL 执行的结果
  2. 思路一:
    //condition
    @Test
    public void test08(){
        //定义查询条件,可能为 null(用户未输入或未选择)
        String name = null;
        Integer ageBegin = 10;
        Integer ageEnd = 24;
        QueryWrapper<User> queryWrapper = new QueryWrapper<>();
        //StringUtils.isNotBlank() 判断某字符串是否不为空且长度不为 0 且不由空白符构成
        if(StringUtils.isNotBlank(name)){
            queryWrapper.like("name","a");
        }
        if(ageBegin != null){
            queryWrapper.ge("age",ageBegin);
        }
        if (ageEnd != null){
            queryWrapper.le("age",ageEnd);
        }
    
        List<User> users = userMapper.selectList(queryWrapper);
        users.forEach(System.out::println);
    }
  3. 思路二:
    1. 上面的实现方案没有问题,但是代码比较复杂,我们可以使用带 comdtion 参数的重载方法构建查询条件,简化代码编写
      @Test
      public void test09(){
          //定义查询条件,有可能为 null(用户未输入或未选择)
          String username = null;
          Integer ageBegin = 10;
          Integer ageEnd = 20;
          QueryWrapper<User> queryWrapper = new QueryWrapper<>();
          queryWrapper.like(StringUtils.isNotBlank(username),"user","a")
                  .ge(ageBegin != null,"age",ageBegin)
                  .le(ageEnd != null,"age",ageEnd);
          List<User> users = userMapper.selectList(queryWrapper);
          users.forEach(System.out::println);
      
      }

LambdaUpdateWrapper:

@Test
public void test10(){
    //组装 set 句子
    LambdaUpdateWrapper<User> updateWrapper = new LambdaUpdateWrapper<>();
    updateWrapper.set(User::getAge,18)
            .set(User::getEmail,"user@qq.com")
            .like(User::getName,"a")    //Lambda 表达式的逻辑优先运算
            .and(i -> i.lt(User::getAge,24).or().isNull(User::getEmail));

    User user = new User();
    int result = userMapper.update(user,updateWrapper);
    System.out.println(result);
}

插件:

  1. 分页插件:
    1. MyBatis-Plus 自带分页插件,只要简单的配置即可实现分页功能
    2. 添加配置类:
      @Configuration
      @MapperScan("com.springboot.mybatisplus.mapper")
      public class MybatisPlusConfig {
          @Bean
          public MybatisPlusInterceptor mybatisPlusInterceptor(){
              MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
              interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));
              return interceptor;
          }
      }
    3. 测试:
      @Test
      public void test11() {
          //设置分页参数
          Page<User> page = new Page<>(1,5);
          userMapper.selectPage(page,null);
          List<User> list = page.getRecords();
          list.forEach(System.out::println);
          System.out.println("当前页:"+page.getCurrent());
          System.out.println("每页显示的条数:"+page.getSize());
          System.out.println("总记录数:"+page.getTotal());
          System.out.println("总页数:"+page.getPages());
          System.out.println("是否有上一页:"+page.hasPrevious());
          System.out.println("是否有下一页:"+page.hasNext());
      }
    4. 测试结果:
  2. xml 自定义分页:
    1. UserMapper 中定义接口方法:
      public interface UserMapper extends BaseMapper<User> {
          /**
           * 根据年龄查询用户列表,分页显示
           * @param page 分页对象,xml 中可以从这里面进行取值,传递参数 Page 即自动分页,必须放在第一位
           * @param age 年龄
           * @return
           */
          Page<User> selectPagecVo(@Param("page")Page<User> page,@Param("age")Integer age);
      }
    2. UserMapper.xml 中编写 SQL:
      <?xml version="1.0" encoding="UTF-8"?>
      <!DOCTYPE mapper
              PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
              "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
      <mapper namespace="com.springboot.mybatisplus.mapper.UserMapper">
      
          <!--SQL 片段,记录基础字段-->
          <sql id="BaseColums">id,username,age,email</sql>
          
          <!--IPage<User> selectPageVo(Page<User> Page,Integer age)-->
          <select id="selectPageVo" resultType="User">
              select <include refid="BaseColums"></include> from t_user where age > #{age}
          </select>
      
      </mapper>
    3. 测试:
      结果:
      User(id=3, name=Tom, age=28, email=test3@baomidou.com, isDeleted=null)
      User(id=4,name=Sandy, age=21, email=test4@baomidou.com, isDeleted=null) 
      User(id=5, name=Billie,age=24, email=test5@baomidou.com, isDeleted=null) 
      User(id=8, name=ybc1, age=21,email=null, isDeleted=null)
      User(id=9, name=ybc2, age=22, email=null, isDeleted=null) 
      当前 页: 1
      每页显示的条数:  5 
      总记录数:   12
      总页数:  3
      是否有上一页: false
      是否有下一页: true

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

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

相关文章

(脚本学习)BUU18 [CISCN2019 华北赛区 Day2 Web1]Hack World1

自用 题目 考虑是不是布尔盲注&#xff0c;如何测试&#xff1a;用"1^1^11 1^0^10&#xff0c;就像是真真真等于真&#xff0c;真假真等于假"这个测试 SQL布尔盲注脚本1 import requestsurl "http://8e4a9bf2-c055-4680-91fd-5b969ebc209e.node5.buuoj.cn…

【Numpy核心编程攻略:Python数据处理、分析详解与科学计算】2.25 多线程并行:GIL绕过与真正并发

2.25 多线程并行&#xff1a;GIL绕过与真正并发 目录 #mermaid-svg-JO4lsTIyjOweVkos {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-JO4lsTIyjOweVkos .error-icon{fill:#552222;}#mermaid-svg-JO4lsTIyjOweVkos …

Java 大视界 -- Java 大数据在智能医疗影像诊断中的应用(72)

💖亲爱的朋友们,热烈欢迎来到 青云交的博客!能与诸位在此相逢,我倍感荣幸。在这飞速更迭的时代,我们都渴望一方心灵净土,而 我的博客 正是这样温暖的所在。这里为你呈上趣味与实用兼具的知识,也期待你毫无保留地分享独特见解,愿我们于此携手成长,共赴新程!💖 一、…

【Leetcode刷题记录】1456. 定长子串中元音的最大数目---定长滑动窗口即解题思路总结

1456. 定长子串中元音的最大数目 给你字符串 s 和整数 k 。请返回字符串 s 中长度为 k 的单个子字符串中可能包含的最大元音字母数。 英文中的 元音字母 为&#xff08;a, e, i, o, u&#xff09;。 这道题的暴力求解的思路是通过遍历字符串 s 的每一个长度为 k 的子串&#xf…

upload-labs安装与配置

前言 作者进行upload-labs靶场练习时&#xff0c;在环境上出了很多问题&#xff0c;吃了很多苦头&#xff0c;甚至改了很多配置也没有成功。 upload-labs很多操作都是旧时代的产物了&#xff0c;配置普遍都比较老&#xff0c;比如PHP版本用5.2.17&#xff08;还有中间件等&am…

从Transformer到世界模型:AGI核心架构演进

文章目录 引言:架构革命推动AGI进化一、Transformer:重新定义序列建模1.1 注意力机制的革命性突破1.2 从NLP到跨模态演进1.3 规模扩展的黄金定律二、通向世界模型的关键跃迁2.1 从语言模型到认知架构2.2 世界模型的核心特征2.3 混合架构的突破三、构建世界模型的技术路径3.1 …

每日一博 - 三高系统架构设计:高性能、高并发、高可用性解析

文章目录 引言一、高性能篇1.1 高性能的核心意义1.2 影响系统性能的因素1.3 高性能优化方法论1.3.1 读优化&#xff1a;缓存与数据库的结合1.3.2 写优化&#xff1a;异步化处理 1.4 高性能优化实践1.4.1 本地缓存 vs 分布式缓存1.4.2 数据库优化 二、高并发篇2.1 高并发的核心意…

【工欲善其事】利用 DeepSeek 实现复杂 Git 操作:从原项目剥离出子版本树并同步到新的代码库中

文章目录 利用 DeepSeek 实现复杂 Git 操作1 背景介绍2 需求描述3 思路分析4 实现过程4.1 第一次需求确认4.2 第二次需求确认4.3 第三次需求确认4.4 V3 模型&#xff1a;中间结果的处理4.5 方案验证&#xff0c;首战告捷 5 总结复盘 利用 DeepSeek 实现复杂 Git 操作 1 背景介绍…

【C++】线程池实现

目录 一、线程池简介线程池的核心组件实现步骤 二、C11实现线程池源码 三、线程池源码解析1. 成员变量2. 构造函数2.1 线程初始化2.2 工作线程逻辑 3. 任务提交(enqueue方法)3.1 方法签名3.2 任务封装3.3 任务入队 4. 析构函数4.1 停机控制 5. 关键技术点解析5.1 完美转发实现5…

数据结构实战之线性表(三)

目录 1.顺序表释放 2.顺序表增加空间 3.合并顺序表 4.线性表之链表实现 1.项目结构以及初始代码 2.初始化链表(不带头结点) 3.链表尾部插入数据并显示 4.链表头部插入数据 5.初始化链表&#xff08;带头结点&#xff09; 6.带头结点的链表头部插入数据并显示 7.带头结…

【python】python基于机器学习与数据分析的手机特性关联与分类预测(源码+数据集)【独一无二】

&#x1f449;博__主&#x1f448;&#xff1a;米码收割机 &#x1f449;技__能&#x1f448;&#xff1a;C/Python语言 &#x1f449;专__注&#x1f448;&#xff1a;专注主流机器人、人工智能等相关领域的开发、测试技术。 python基于机器学习与数据分析的手机特性关联与分类…

ZOJ 1007 Numerical Summation of a Series

原题目链接 生成该系列值的表格 对于x 的 2001 个值&#xff0c;x 0.000、0.001、0.002、…、2.000。表中的所有条目的绝对误差必须小于 0.5e-12&#xff08;精度为 12 位&#xff09;。此问题基于 Hamming (1962) 的一个问题&#xff0c;当时的大型机按今天的微型计算机标准来…

全面解析文件上传下载删除漏洞:风险与应对

在数字化转型的时代&#xff0c;文件上传、下载与删除功能已经成为各类应用程序的标准配置&#xff0c;从日常办公使用的协同平台&#xff0c;到云端存储服务&#xff0c;再到社交网络应用&#xff0c;这些功能在给用户带来便捷体验、显著提升工作效率的同时&#xff0c;也隐藏…

【C语言深入探索】结构体详解(二):使用场景

目录 一、复杂数据的表示 二、数据的封装 三、多态的模拟 四、回调函数的实现 五、多线程编程 六、通信协议的实现和文件操作 6.1. 使用结构体实现简单通信协议 6.2. 使用结构体进行文件操作 七、图形界面编程 结构体在C语言中具有广泛的应用场景&#xff0c;以下是一…

【大模型】AI 辅助编程操作实战使用详解

目录 一、前言 二、AI 编程介绍 2.1 AI 编程是什么 2.1.1 为什么需要AI辅助编程 2.2 AI 编程主要特点 2.3 AI编程底层核心技术 2.4 AI 编程核心应用场景 三、AI 代码辅助编程解决方案 3.1 AI 大模型平台 3.1.1 AI大模型平台代码生成优缺点 3.2 AI 编码插件 3.3 AI 编…

RK3566-移植5.10内核Ubuntu22.04

说明 记录了本人使用泰山派&#xff08;RK3566&#xff09;作为平台并且成功移植5.10.160版本kernel和ubuntu22.04&#xff0c;并且成功配置&连接网络的完整过程。 本文章所用ubuntu下载地址&#xff1a;ubuntu-cdimage-ubuntu-base-releases-22.04-release安装包下载_开源…

从零开始实现一个双向循环链表:C语言实战

文章目录 1链表的再次介绍2为什么选择双向循环链表&#xff1f;3代码实现&#xff1a;从初始化到销毁1. 定义链表节点2. 初始化链表3. 插入和删除节点4. 链表的其他操作5. 打印链表和判断链表是否为空6. 销毁链表 4测试代码5链表种类介绍6链表与顺序表的区别7存储金字塔L0: 寄存…

51单片机 06 定时器

51 单片机的定时器属于单片机的内部资源&#xff0c;其电路的连接和运转均在单片机内部完成。 作用&#xff1a;1、用于计时&#xff1b;2、替代长时间的Delay&#xff0c;提高CPU 运行效率和处理速度。 定时器个数&#xff1a;3个&#xff08;T0、T1、T2&#xff09;&#xf…

【C++】P1957 口算练习题

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 &#x1f4af;前言&#x1f4af;题目描述输入格式&#xff1a;输出格式&#xff1a; &#x1f4af;我的做法代码实现&#xff1a; &#x1f4af;老师的做法代码实现&#xff1a; &#x1f4af;对比分析&am…

Workbench 中的热源仿真

探索使用自定义工具对移动热源进行建模及其在不同行业中的应用。 了解热源动力学 对移动热源进行建模为各种工业过程和应用提供了有价值的见解。激光加热和材料加工使用许多激光束来加热、焊接或切割材料。尽管在某些情况下&#xff0c;热源 &#xff08;q&#xff09; 不是通…