MyBatisPlus 之三:BaseMapper 详解和 CRUD 案例详解

2024-03-18 16:04

本文主要是介绍MyBatisPlus 之三:BaseMapper 详解和 CRUD 案例详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

BaseMapper详解

1. SQL 日志开启

为了更好更快学习 MyBatisPlus ,需要配置 SQL 日志,这样方便我们能随时看到执行过程中使用的 SQL 语句,有助于理解执行原理及方便 SQL 错误调试

mybatis-plus.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImplmybatis-plus.configuration.map-underscore-to-camel-case=false 

2. BaseMapper API

BaseMapper 是 MyBatis Plus 提供的一个通用接口,它定义了一套基础的 CRUD(Create, Read, Update, Delete)操作方法。通过继承 BaseMapper,开发者可以直接在自定义的 Mapper 接口中获得这些基本的数据操作功能,而无需手动编写 SQL 语句或 XML 映射文件。

public interface BaseMapper<T> extends Mapper<T> {int insert(T entity);int deleteById(Serializable id);int deleteByMap(@Param("cm") Map<String, Object> columnMap);int delete(@Param("ew") Wrapper<T> queryWrapper);int deleteBatchIds(@Param("coll") Collection<? extends Serializable> idList);int updateById(@Param("et") T entity);int update(@Param("et") T entity, @Param("ew") Wrapper<T> updateWrapper);T selectById(Serializable id);List<T> selectBatchIds(@Param("coll") Collection<? extends Serializable> idList);List<T> selectByMap(@Param("cm") Map<String, Object> columnMap);T selectOne(@Param("ew") Wrapper<T> queryWrapper);Integer selectCount(@Param("ew") Wrapper<T> queryWrapper);List<T> selectList(@Param("ew") Wrapper<T> queryWrapper);List<Map<String, Object>> selectMaps(@Param("ew") Wrapper<T> queryWrapper);List<Object> selectObjs(@Param("ew") Wrapper<T> queryWrapper);<E extends IPage<T>> E selectPage(E page, @Param("ew") Wrapper<T> queryWrapper);<E extends IPage<Map<String, Object>>> E selectMapsPage(E page, @Param("ew") Wrapper<T> queryWrapper);
}

3 . 基本CRUD

1) 插入数据:
int insert(T entity);
/*** 1. 如果属性使用的是驼峰命名法时,生成SQL时,转为带下划线的列名。例如:bookName--book_name*   解决办法:*   a. 给属性添加注解 @TableField("bookName")*   b. 在配置文件中,关闭驼峰名和下划线的转换*/
@Test
void save(){Book book = new Book();book.setPrice(34f);book.setAuthor("王超");book.setBookName("JSP开发技术");book.setPubDate(new Date());System.out.println(book);bookDao.insert(book);System.out.println(book);   //执行前 后 观察变化。主键
}

注意:

  1. 如果属性使用的是驼峰命名法时,生成SQL时,转为带下划线的列名。例如:bookName--book_name

    解决办法:

    a. 给属性添加注解 @TableField("bookName")

    b. 在配置文件中,关闭驼峰名和下划线的转换

    mybatis-plus.configuration.map-underscore-to-camel-case=false
  2. 如果指定了主键 @TableId 后,默认情况下(没有指定主键生成策略),主键会使用雪花算法生成应用于分布式系统的主键

    解决办法:

    a. 给主键指定生成策略 @TableId(value = "bookid",type = IdType.AUTO)

    b. IdType.AUTO : 主键生成有数据库端来完成,一般适用于数据库自动增长

type = IdType.INPUT :程序自己指定主键
type = IdType.ASSIGN_ID :由框架自动生成指定的主键,使用的是雪花算法
type = IdType.ASSIGN_UUID :有框架自动生成给定主键,使用算法 UUID 算法

除了AUTO 之外,都是程序指定的主键,又区分程序员自己指定还是框架自动生成。

2) 查询:
T selectById(Serializable id);//根据指定的主键列表查询
List<T> selectBatchIds(@Param("coll") Collection<? extends Serializable> idList);
//根据map中的key-value 作为条件等值条件查询
List<T> selectByMap(@Param("cm") Map<String, Object> columnMap);
//根据构建的条件查询唯一一条记录对象
T selectOne(@Param("ew") Wrapper<T> queryWrapper);//根据构建的条件 查询记录数
Integer selectCount(@Param("ew") Wrapper<T> queryWrapper);
//根据构建的条件 查询多条记录
List<T> selectList(@Param("ew") Wrapper<T> queryWrapper);List<Map<String, Object>> selectMaps(@Param("ew") Wrapper<T> queryWrapper);List<Object> selectObjs(@Param("ew") Wrapper<T> queryWrapper);<E extends IPage<T>> E selectPage(E page, @Param("ew") Wrapper<T> queryWrapper);<E extends IPage<Map<String, Object>>> E selectMapsPage(E page, @Param("ew") Wrapper<T> queryWrapper);

如果直接使用 selectById() 查询,结果下面语句:

/*** 注意:给实体类中指定主键,否则出现下面 null=?* SELECT bookId,bookName,author,pubdate,price FROM book WHERE null=?*/

查不出结果的原因:不知道哪一个列是主键。所以使用了 null = ? 作为条件了

解决办法:

给属性指定主键:@TableId

@TableField 可以设置忽略 exist=true

@Data
public class Book {@TableIdprivate Integer bookId;@TableField("bookName")private String bookName;private Float price;@TableField("pubdate")private Date pubDate;private String author;
}
selectBatchIds() 用法:
//SELECT bookid,bookName,price,pubdate,author FROM book WHERE bookid IN ( ? , ? , ? , ? , ? )
List<Integer> ids = Arrays.asList(2,3,4,26,27);
bookDao.selectBatchIds(ids);
selectByMap() 用法:
//SELECT bookid,bookName,price,pubdate,author FROM book WHERE price = ? AND bookname = ?
Map<String,Object> cond = new HashMap<>();
cond.put("price",34f);
cond.put("bookname","Java开发技术");bookDao.selectByMap(cond);
Wrapper 用法

下面方法用法:Wrapper 用法

//根据构建的条件 查询记录数
Integer selectCount(@Param("ew") Wrapper<T> queryWrapper);
//根据构建的条件 查询多条记录
List<T> selectList(@Param("ew") Wrapper<T> queryWrapper);

下面使用方式:

QueryWrapper 可以用于构建查询的列、条件、排序、分组等要素。

    void queryByWrapper(){QueryWrapper<Book> queryWrapper = new QueryWrapper<>();queryWrapper.select("bookname","price","author");     //指定要求查询的列queryWrapper.gt("price",20);                            // where price > 20queryWrapper.le("price",80);                            // and  price <= 80queryWrapper.eq("bookname","Java开发实战");             // and bookname= 'Java开发实战'queryWrapper.between("bookid",10,50);                // and bookid between 10,50 bookDao.selectList(queryWrapper);//       bookDao.selectCount(null); //无条件查询,没有where
//        bookDao.selectCount(queryWrapper); //按条件查询}

queryWrapper 支持链式编程方式:

queryWrapper.select("bookname","price","author")
.gt("price",20)
.le("price",80)
.eq("bookname","Java开发实战")
.between("bookid",10,50);
bookDao.selectList(queryWrapper);
selectOne()

注意:返回值只能小于等于一条才可以,否则异常。

@Test
void queryOne(){QueryWrapper<Book> queryWrapper = new QueryWrapper<>();queryWrapper.eq("bookname", "Java实战开发1").eq("author","周瑜");Book book = bookDao.selectOne(queryWrapper);System.out.println(book);
}
selectMaps()

返回结果为List ,元素为Map封装的记录对象。

@Test
void queryMap(){List<Map<String, Object>> maps = bookDao.selectMaps(null);maps.forEach(System.out::println);
}
or 和 orderby
/*** where author='周瑜' or author='诸葛亮'** or 和 order by*/
@Test
public void or(){QueryWrapper<Book> queryWrapper = new QueryWrapper<>();queryWrapper.eq("author","周瑜").or().eq("author","诸葛亮").or().gt("price",30f).le("price",60f); // 默认为and 关系queryWrapper.orderByAsc("price").orderByDesc("bookid");bookDao.selectList(queryWrapper);
}
分页

分页需要考虑的要素:每页行数、总记录数、总页数、当前页号、当前页要显示记录

不同数据库分页 SQL 不同。

mysql : limit start, count;

sqlserver : top ---not in

oracle : rownum

MyBatisPlus 会根据 “方言” 配置,自动生成对应数据库的 SQL 语句去执行。

代码如下:

@Test
void page(){Page<Book> page = new Page<>();page.setSize(5);    //每页显示的记录数page.setCurrent(3); //设置当前页       // limit startindex ,countpage//(curPage-1)*size Page<Book> page1 = bookDao.selectPage(page, null);//当前的要显示的记录List<Book> records = page1.getRecords();System.out.println(page1==page);records.forEach(System.out::println);
}

注意:直接运行后,发现并没有实现分页。

原因:没有安装分页插件。

解决办法:去官网,查看分页插件使用方法。复制代码到项目中即可

https://mp.baomidou.com/guide/page.html

//Spring boot方式
@Configuration
//@MapperScan("com.baomidou.cloud.service.*.mapper*")
public class MybatisPlusConfig {// 旧版
//    @Beanpublic PaginationInterceptor paginationInterceptor() {PaginationInterceptor paginationInterceptor = new PaginationInterceptor();// 设置请求的页面大于最大页后操作, true调回到首页,false 继续请求  默认false// paginationInterceptor.setOverflow(false);// 设置最大单页限制数量,默认 500 条,-1 不受限制// paginationInterceptor.setLimit(500);// 开启 count 的 join 优化,只针对部分 left joinpaginationInterceptor.setCountSqlParser(new JsqlParserCountOptimize(true));return paginationInterceptor;}// 最新版@Beanpublic MybatisPlusInterceptor mybatisPlusInterceptor() {MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL)); //数据库方言return interceptor;}}

2023最新版本的分页插件:

package com.wdzl.plus.config;import com.baomidou.mybatisplus.annotation.DbType;
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 MyBatisPlusConfig {@Beanpublic MybatisPlusInterceptor mybatisPlusInterceptor() {MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));return interceptor;}
}

注意:

  1. 修改数据库方言(类型)

  2. 使用新版。注释掉旧版,不能同时使用。

  3. 类名和方法名可以自定义

另外:上面测试代码中

可以看到 page1==page 的输出结果为 true,说明两个是同一个对象。可以直接使用page 即可,简化后:

@Test
void page(){Page<Book> page = new Page<>();page.setSize(5);    //每页显示的记录数page.setCurrent(3); //设置当前页       // limit startindex ,countpagebookDao.selectPage(page, null);//当前的要显示的记录List<Book> records = page.getRecords();records.forEach(System.out::println);
}
3)修改
    //update book set price=99 where bookid=?int updateById(@Param("et") T entity);//update book set price=99 where bookname=? and price<?int update(@Param("et") T entity, @Param("ew") Wrapper<T> updateWrapper);
    @Testpublic void update(){
//        Book book = bookDao.selectById(27);
//        book.setPrice(88f);
//        bookDao.updateById(book); //按主键修改//下面是按条件修改QueryWrapper<Book> queryWrapper = new QueryWrapper<>();queryWrapper.eq("price",88);//注意 :book参数 ,是作为修改后的目标数据。把符合条件的所有记录都修改为book相同的值。Book book1 = new Book();book1.setPrice(100f);bookDao.update(book1,queryWrapper);//update 和 updateById() 在修改时,只修改属性不为null的列。}

注意:

  1. 单个对象修改,一般是先查询数据,再修改个别列。
  2. 对应null 的属性,不会被修改,不会执行SQL ,只会把有值的属性生成到SQL中。
4)删除
    //按主键删除 自注释int deleteById(Serializable id);//按map中的key-value 等值条件 删除int deleteByMap(@Param("cm") Map<String, Object> columnMap);// 根据构建条件删除int delete(@Param("ew") Wrapper<T> queryWrapper);// 根据集合中主键列表删除int deleteBatchIds(@Param("coll") Collection<? extends Serializable> idList);
    @Testvoid delete(){Map<String,Object> map = new HashMap<>();map.put("price",44f);map.put("bookname","java开发");
//        bookDao.deleteByMap(map);//WHERE bookid IN ( ? , ? , ? , ? , ? , ? )List<Integer> ids = Arrays.asList(3,4,5,22,34,55);
//        bookDao.deleteBatchIds(ids);QueryWrapper<Book> queryWrapper = new QueryWrapper<>();queryWrapper.lt("price",30);bookDao.delete(queryWrapper);

这篇关于MyBatisPlus 之三:BaseMapper 详解和 CRUD 案例详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/822875

相关文章

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

Python中注释使用方法举例详解

《Python中注释使用方法举例详解》在Python编程语言中注释是必不可少的一部分,它有助于提高代码的可读性和维护性,:本文主要介绍Python中注释使用方法的相关资料,需要的朋友可以参考下... 目录一、前言二、什么是注释?示例:三、单行注释语法:以 China编程# 开头,后面的内容为注释内容示例:示例:四

mysql表操作与查询功能详解

《mysql表操作与查询功能详解》本文系统讲解MySQL表操作与查询,涵盖创建、修改、复制表语法,基本查询结构及WHERE、GROUPBY等子句,本文结合实例代码给大家介绍的非常详细,感兴趣的朋友跟随... 目录01.表的操作1.1表操作概览1.2创建表1.3修改表1.4复制表02.基本查询操作2.1 SE

MySQL中的锁机制详解之全局锁,表级锁,行级锁

《MySQL中的锁机制详解之全局锁,表级锁,行级锁》MySQL锁机制通过全局、表级、行级锁控制并发,保障数据一致性与隔离性,全局锁适用于全库备份,表级锁适合读多写少场景,行级锁(InnoDB)实现高并... 目录一、锁机制基础:从并发问题到锁分类1.1 并发访问的三大问题1.2 锁的核心作用1.3 锁粒度分

MySQL数据库中ENUM的用法是什么详解

《MySQL数据库中ENUM的用法是什么详解》ENUM是一个字符串对象,用于指定一组预定义的值,并可在创建表时使用,下面:本文主要介绍MySQL数据库中ENUM的用法是什么的相关资料,文中通过代码... 目录mysql 中 ENUM 的用法一、ENUM 的定义与语法二、ENUM 的特点三、ENUM 的用法1

Python中re模块结合正则表达式的实际应用案例

《Python中re模块结合正则表达式的实际应用案例》Python中的re模块是用于处理正则表达式的强大工具,正则表达式是一种用来匹配字符串的模式,它可以在文本中搜索和匹配特定的字符串模式,这篇文章主... 目录前言re模块常用函数一、查看文本中是否包含 A 或 B 字符串二、替换多个关键词为统一格式三、提

MySQL count()聚合函数详解

《MySQLcount()聚合函数详解》MySQL中的COUNT()函数,它是SQL中最常用的聚合函数之一,用于计算表中符合特定条件的行数,本文给大家介绍MySQLcount()聚合函数,感兴趣的朋... 目录核心功能语法形式重要特性与行为如何选择使用哪种形式?总结深入剖析一下 mysql 中的 COUNT

一文详解Git中分支本地和远程删除的方法

《一文详解Git中分支本地和远程删除的方法》在使用Git进行版本控制的过程中,我们会创建多个分支来进行不同功能的开发,这就容易涉及到如何正确地删除本地分支和远程分支,下面我们就来看看相关的实现方法吧... 目录技术背景实现步骤删除本地分支删除远程www.chinasem.cn分支同步删除信息到其他机器示例步骤

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

MyBatisPlus如何优化千万级数据的CRUD

《MyBatisPlus如何优化千万级数据的CRUD》最近负责的一个项目,数据库表量级破千万,每次执行CRUD都像走钢丝,稍有不慎就引起数据库报警,本文就结合这个项目的实战经验,聊聊MyBatisPl... 目录背景一、MyBATis Plus 简介二、千万级数据的挑战三、优化 CRUD 的关键策略1. 查