mybatis-plus使用记录

news2025/7/26 19:49:13

MyBatis-Plus 学习笔记

一、 快速入门

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

1. 引入 Maven 依赖

要使用 MyBatis-Plus,首先需要在项目的 pom.xml 文件中引入相关依赖。

官网坐标查看:https://baomidou.com/getting-started/install/

代码段

<dependency>
    <groupId>com.baomidou</groupId>
    <artifactId>mybatis-plus-boot-starter</artifactId>
    <version>最新版本</version>
</dependency>

2. Mapper 层开发

与 MyBatis 不同,使用 MP 后,Mapper 接口只需继承 BaseMapper 即可获得强大的 CRUD 功能,无需编写基础的 SQL 语句。

  • 创建 Mapper 接口: 让你的 Mapper 接口继承 BaseMapper<T>
  • 指定泛型: 泛型 T 必须是与数据库表对应的实体类(Entity)。

Java

@Mapper
public interface UserMapper extends BaseMapper<User> {
    // 复杂的、自定义的SQL可以写在这里
}

3. Service 层开发

在 Service 层,通过 Spring 的依赖注入(如 @Autowired)引入 Mapper,然后就可以直接调用继承自 BaseMapper 的各种方法。

Java

@Service
public class UserServiceImpl implements IUserService {

    @Autowired
    private UserMapper userMapper;

    public User getUserById(Long id) {
        // 直接调用 BaseMapper 提供的方法
        return userMapper.selectById(id);
    }
}

二、 核心概念与配置

1. 实体类与表映射

默认原则:

MP 默认采用驼峰命名与下划线命名的映射规则。 例如,Java 实体类的 userName 字段会自动映射到数据库表的 user_name 字段。

手动映射注解:

当实体类属性名与数据库字段名不一致,或有特殊需求时,可以使用注解进行手动映射。

  • @TableName: 指定实体类对应的表名。
  • @TableId: 声明该字段为主键。
    • type = IdType.AUTO: 设置为主键自增。如果不设置,MP 默认会使用“雪花算法”生成一个长整型的 ID。
  • @TableField: 声明非主键的字段。
    • value = "db_column_name": 指定映射的数据库字段名。
    • exist = false: 表示该字段在数据库表中不存在,只是业务逻辑需要。

何时需要使用TableField:

Java

@Data
@TableName("sys_user") // 指定表名
public class User {
    @TableId(type = IdType.AUTO) // 主键自增
    private Long id;

    @TableField("user_name") // 字段名映射
    private String name;

    private Integer age;

    @TableField(exist = false) // 数据库中无此字段
    private String gender;
}

2. 常见配置

可以在 application.ymlapplication.properties 中添加 MP 的相关配置。

YAML

mybatis-plus:
  type-aliases-package: com.example.project.domain # 实体类别名扫描包
  mapper-locations: classpath*:/mapper/**/*.xml   # Mapper XML 文件位置
  configuration:
    map-underscore-to-camel-case: true           # 开启驼峰下划线转换
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl # 打印SQL日志

三、 条件构造器 (Wrapper)

条件构造器是 MP 的精髓所在,它能让我们以面向对象的方式构建复杂的 SQL 查询条件,而无需拼接字符串。

1. QueryWrapper

用于构建查询条件(SELECT 语句的 WHERE 部分)。

Java

// 查询年龄大于等于20,且状态为1的用户
QueryWrapper<User> queryWrapper = new QueryWrapper<>();
queryWrapper.ge("age", 20)
            .eq("status", 1)
            .orderByDesc("create_time");

List<User> users = userMapper.selectList(queryWrapper);

2. UpdateWrapper

主要用于构建更新条件,可以精细化控制 UPDATE 语句的 SETWHERE 部分。

Java

// 将所有状态为1的用户的年龄更新为18岁
UpdateWrapper<User> updateWrapper = new UpdateWrapper<>();
updateWrapper.set("age", 18)      // 设置要更新的字段和值
             .eq("status", 1);  // 设置更新条件

userMapper.update(null, updateWrapper); // entity传null,表示不使用实体对象更新

3. Lambda 版 Wrapper

LambdaQueryWrapperLambdaUpdateWrapper 是更推荐的使用方式。它们利用 Java 8 的 Lambda 表达式,可以避免硬编码字段名字符串,提高了代码的可读性和安全性。

  • 传统方式: .eq("status", 1)
  • Lambda方式: .eq(User::getStatus, 1)

Java

// 使用 LambdaUpdateWrapper
userMapper.update(null, 
    new LambdaUpdateWrapper<User>()
        .set(User::getAge, 18)
        .eq(User::getStatus, 1)
);

4.自定义sql:

当where条件前面的内容比较难写,需要拼接的时候,用该方式,调用的是自定义的方法

四、 Service 层封装

MP 对 IService 接口也提供了默认实现 ServiceImpl,进一步简化开发。

1. 接口与实现

  • 接口: 业务接口继承 IService<T>
  • 实现类: 继承 ServiceImpl<M, T>,并实现自己的业务接口。

Java

// 接口 IUserService.java
public interface IUserService extends IService<User> {
    // 自定义复杂业务方法
    PageResult<UserDTO> getUserListPage(UserQuery query);
}

// 实现类 UserServiceImpl.java
@Service
public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IUserService {
    
    // ServiceImpl已经注入了baseMapper,可以直接使用
    
    // 实现自定义方法
    @Override
    public PageResult<UserDTO> getUserListPage(UserQuery query) {
        // ... 复杂逻辑
        long total = count(); // 调用IService提供的方法
        List<User> users = baseMapper.selectUserWithDetails(query); // 调用Mapper自定义方法
        // ...
    }
}

2. 常用方法

ServiceImpl 提供了一系列便捷的单表操作方法:

  • getOne(wrapper): 查询单个对象。
  • list(wrapper): 查询对象列表。
  • count(wrapper): 查询总记录数。
  • save(entity): 新增。
  • updateById(entity): 根据 ID 更新。
  • removeById(id): 根据 ID 删除。

使用示例: 在 Controller 中注入 IUserService 后,可以直接调用这些方法。

五、 分页查询

MP 内置了强大的分页插件,使用非常简单。

1. 添加分页插件配置

需要将分页插件作为一个 Spring Bean 进行配置。

Java

@Configuration
public class MybatisPlusConfig {

    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {
        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
        // 添加分页插件
        interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));
        return interceptor;
    }
}

2. 在 Service/Controller 中使用

分页查询的核心是构建一个 Page 对象。

Java

public Page<User> findUsersByPage(int pageNum, int pageSize) {
    // 1. 创建 Page 对象,传入当前页码和每页数量
    Page<User> page = new Page<>(pageNum, pageSize);

    // 2. 调用 Mapper 的 selectPage 方法
    // MP 会自动拦截 SQL,并拼接上 LIMIT 分页语句
    userMapper.selectPage(page, null); // wrapper为null表示查询所有

    // 3. page 对象中就包含了分页数据和总数
    return page;
}

查询结果 Page<T> 对象包含了丰富的分页信息,如:

  • getRecords(): 获取当前页的数据列表。
  • getTotal(): 获取总记录数。
  • getPages(): 获取总页数。
  • getCurrent(): 获取当前页码。
  • getSize(): 获取每页显示数量。

六、使用流程

决策流程:由上至下的选择策略

您可以将其视为一个决策层级,永远从最顶层(最抽象、最方便)的方法开始尝试,只有当前层级无法满足需求时,才往下一個层级移动。

  1. 首先,考虑 **IService** 的内置方法:这个功能是否为一个简单的单表 CRUD?
  2. 如果不是,考虑自定义 **Service** 方法:这个功能是否为一个复杂的业务流程(例如多个步骤、需要事务控制、数据转换)?
  3. **在自定义 **Service** 方法中,考虑 ****BaseMapper**:我是否需要调用 Mapper 层的方法来执行数据库操作?
  4. 最后,考虑自定义 **Mapper** 方法 + SQL:这个查询是否非常复杂(例如多表 JOIN),以至于 Wrapper 无法满足?

1. 何时使用 IService 中封装的方法(默认首选)

这是指 ServiceImpl 中已经为您准备好的方法,如 save(), getById(), removeById(), list(), page(), count() 等。

  • 核心用途:处理所有标准的、单一数据表的“增、删、改、查”(CRUD) 操作。
  • 使用时机
    • 当您执行的只是一个基础的、针对单一数据表的数据操作时。
    • 查询条件可以用 QueryWrapperLambdaQueryWrapper 轻松构建。
    • 这应该是您的第一选择与默认方法,通常由 Controller 层直接调用,或在简单的业务方法中被使用。

示例:在 Controller 中根据 ID 获取用户信息。

  • Java
@GetMapping("/{id}")
public UserDTO findUserById(@PathVariable Long id) {
    // IService 的 getById 是最完美的选择,因为这是一个简单的单表查询
    User user = userService.getById(id); //
    // 可能会搭配 hutool 做对象转换
    return BeanUtil.copyProperties(user, UserDTO.class); //
}

2. 何时使用自定义 Service 方法(业务逻辑层)

这是在您自己的 IUserService 接口中定义,并在 UserServiceImpl 中实现的方法。

  • 核心用途封装复杂的业务逻辑,让 Controller 层保持简洁。一个业务逻辑通常不仅仅是一次数据库查询。
  • 使用时机
    • 当一个功能需要多个步骤才能完成时(例如:先新增订单,再更新库存)。
    • 当方法需要加入事务控制 (@Transactional) 时。
    • 当逻辑中包含非数据库操作时(例如:调用外部 API、发送邮件、文件处理)。
    • 当需要进行数据转换与组合时(例如:查询多个表,将结果组合成一个视图对象 DTO/VO)。
    • 当您想将一段复杂的查询逻辑重复使用时。

示例:一个复杂的分页查询,它不仅需要查询数据,还可能需要处理查询参数、调用 Mapper 进行复杂查询,并对结果进行封装。

  • Java
// 在 IUserService.java 中定义
public interface IUserService extends IService<User> {
    PageResult<User> getUserList(QueryParams params);
}

// 在 UserServiceImpl.java 中实现
@Service
public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IUserService {
    @Override
    public PageResult<User> getUserList(QueryParams params) {
        // 1. 处理业务逻辑...

        // 2. 调用 Mapper 层进行数据库操作
        List<User> users = baseMapper.selectUserWithDetails(params); //

        // 3. 调用 IService 的内置方法获取总数
        long total = count(); //

        // 4. 封装结果并返回
        return new PageResult<>(users, total);
    }
}

3. 何时使用 BaseMapper 中的方法(基础数据访问)

这是 IService 底层实现所依赖的基础 CRUD 方法,例如 selectById(), selectList(), insert() 等。

  • 核心用途:作为 ServiceImpl 与 Mapper 层沟通的桥梁。
  • 使用时机
    • 在现代 MyBatis-Plus 开发中,您很少会直接使用 **baseMapper** 的内置方法(如 baseMapper.selectById()),因为在 ServiceImpl 中可以直接使用 this.getById(),后者更方便且功能一致。
    • baseMapper 对象最主要的用途是:在自定义的 **Service** 方法中,调用您在 **Mapper** 层自定义的方法(即下一点描述的情况)。
  • 示例:如上一个例子所示,baseMapper.selectUserWithDetails(params) 就是通过 baseMapper 来调用在 UserMapper.xml 中自定义的 SQL 查询。

4. 何时使用自定义 Mapper 方法并自定义 SQL(终极武器)

这是在您的 UserMapper 接口中定义一个新方法,并在对应的 XML 文件中编写其完整的 SQL 语句。

  • 核心用途:执行任何 Wrapper 产生不了的复杂 SQL 查询,提供对 SQL 的完全控制权。
  • 使用时机(当 **Wrapper** 不敷使用时)
    • 需要进行多表 **JOIN** 关联查询时。
    • 需要使用复杂的子查询UNION 操作时。
    • 需要使用特定数据库的函数时(例如 MySQL 的 JSON_EXTRACTFIND_IN_SET)。
    • 有非常复杂的 GROUP BYHAVING 条件时。
    • 对于性能要求极高,需要手动优化和调整 SQL 的场景。

示例:在 UserMapper.xml 中定义一个关联查询。

  • XML
<select id="selectUserWithDetails" resultType="com.example.UserVO">
    SELECT
        u.id,
        u.name,
        d.dept_name
    FROM
        user u
    LEFT JOIN
        department d ON u.dept_id = d.id
    WHERE
        u.status = #{params.status}
</select>

然后在 ServiceImpl 中通过 baseMapper 调用它。

总结对照表

方法类型核心用途使用时机示例
**IService**
** 内置方法**
标准、单表 CRUD默认首选。所有简单的单表操作。userService.getById(1L);
自定义 **Service**
方法
封装复杂业务逻辑涉及多步骤、事务、外部调用、数据转换等。userService.createUserAndSendEmail(user);
**BaseMapper**
** 内置方法**
IService
的底层实现
较少直接用,主要用于在 Service 中调用自定义 Mapper 方法baseMapper.selectUserWithDetails(params);
自定义 **Mapper**
方法 + SQL
处理 Wrapper
无法应对的复杂 SQL
最后手段。多表 JOIN、复杂子查询、SQL 优化。在 XML 中写 JOIN
查询。

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

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

相关文章

Mcu_Bsdiff_Upgrade

系统架构 概述 MCU BSDiff 升级系统通过使用二进制差分技术&#xff0c;提供了一种在资源受限的微控制器上进行高效固件更新的机制。系统不传输和存储完整的固件映像&#xff0c;而是只处理固件版本之间的差异&#xff0c;从而显著缩小更新包并降低带宽要求。 该架构遵循一个…

有监督学习——决策树

任务 1、基于iris_data.csv数据&#xff0c;建立决策树模型&#xff0c;评估模型表现; 2、可视化决策树结构; 3、修改min_samples_leaf参数&#xff0c;对比模型结果 代码工具&#xff1a;jupyter notebook 参考资料 20.23 决策树&#xff08;1&#xff09;_哔哩哔哩_bil…

华为OD机试真题——启动多任务排序(2025B卷:200分)Java/python/JavaScript/C/C++/GO最佳实现

2025 B卷 200分 题型 本专栏内全部题目均提供Java、python、JavaScript、C、C++、GO六种语言的最佳实现方式; 并且每种语言均涵盖详细的问题分析、解题思路、代码实现、代码详解、3个测试用例以及综合分析; 本文收录于专栏:《2025华为OD真题目录+全流程解析+备考攻略+经验分…

大模型时代,Python 近红外光谱与 Transformer 模型:学习的必要性探究

在当下大语言模型盛行的时代&#xff0c;各类新技术如潮水般不断涌现&#xff0c;让人应接不暇。身处这样的浪潮之中&#xff0c;不少人心中都会泛起疑问&#xff1a;Python 近红外光谱和 Transformer 模型还有学习的必要性吗&#xff1f;今天&#xff0c;就让我们深入探讨一番…

梯度优化提示词:精准引导AI分类

基于梯度优化的提示词工程方法,通过迭代调整提示词的嵌入向量,使其能够更有效地引导模型做出正确分类。 数据形式 训练数据 train_data 是一个列表,每个元素是一个字典,包含两个键: text: 需要分类的文本描述label: 对应的标签(“冲动"或"理性”)示例数据: …

AUTOSAR 运行时环境 (RTE)

目录 往期推荐 什么是运行时环境&#xff1f; AUTOSAR 中的运行时环境 (RTE) RTE 的应用 RTE 的生成 关于RTE API的一些信息 RTE生成后文件之间的关系 往期推荐 2025汽车行业新宠&#xff1a;欧企都在用的工具软件ETAS工具链自动化实战指南&#xff1c;一&#xff1e;ET…

SQLMesh 宏操作符详解:提升 SQL 查询的灵活性与效率

SQLMesh 提供了一系列强大的宏操作符&#xff08;如 WITH、JOIN、WHERE 等&#xff09;&#xff0c;用于动态构建 SQL 查询。这些操作符不仅简化了复杂查询的编写&#xff0c;还提高了代码的可读性和可维护性。本文将深入探讨这些操作符的使用场景、语法及实际案例&#xff0c;…

基于Flink的数据中台管理平台

基于Flink做的数据中台工程项目。数据从source到clickhouse全流程的验证。集成元数据管、数据资产、数据发现功能&#xff0c;自主管理元数据变更&#xff0c;集成元数据版本管理。 同时&#xff0c;对整个大数据集群使用到的组件或者是工具进行管理。比如nacos、kafka、zookee…

AI-Ready TapData:如何基于 MCP 协构建企业级 AI 实时数据中枢?(含教程)

随着企业对私有大模型、行业大模型的探索逐渐深入&#xff0c;“AI应用是否真正落地”&#xff0c;越来越取决于企业是否拥有结构化、实时、可交互的高质量数据。而现实是&#xff0c;大多数企业的核心业务数据依旧被困在多个异构系统、孤岛数据库和 ETL 流程之中&#xff0c;导…

Spring Boot 登录实现:JWT 与 Session 全面对比与实战讲解

Spring Boot 登录实现&#xff1a;JWT 与 Session 全面对比与实战讲解 2025.5.21-23:11今天在学习黑马点评时突然发现用的是与苍穹外卖jwt不一样的登录方式-Session&#xff0c;于是就想记录一下这两种方式有什么不同 在实际开发中&#xff0c;登录认证是后端最基础也是最重要…

SpringBoot 项目实现操作日志的记录(使用 AOP 注解模式)

本文是博主在做关于如何记录用户操作日志时做的记录&#xff0c;常见的项目中难免存在一些需要记录重要日志的部分&#xff0c;例如权限和角色设定&#xff0c;重要数据的操作等部分。 博主使用 Spring 中的 AOP 功能&#xff0c;结合注解的方式&#xff0c;对用户操作过的一些…

AI|Java开发 IntelliJ IDEA中接入本地部署的deepseek方法

目录 连接本地部署的deepseek&#xff1a; IntelliJ IDEA中使用deepseek等AI&#xff1a; 用法一&#xff1a;让AI写代码 用法二&#xff1a;选中这段代码&#xff0c;右键&#xff0c;可以让其解释这段代码的含义。这时显示的解释是英文的。 连接本地部署的deepseek&#…

【1——Android端添加隐私协议(unity)1/3】

前言&#xff1a;这篇仅对于unity 发布Android端上架国内应用商店添加隐私协议&#xff0c;隐私协议是很重要的东西&#xff0c;没有这个东西&#xff0c;是不上了应用商店的。 对于仅仅添加隐私协议&#xff0c;我知道有三种方式,第一种和第二种基本一样 1.直接在unity里面新…

Linux之概述和安装vm虚拟机

文章目录 操作系统概述硬件和软件操作系统常见操作系统 初识LinuxLinux的诞生Linux内核Linux发行版 虚拟机介绍虚拟机 VMware WorkStation安装虚拟化软件VMware WorkStation 安装查看VM网络连接设置VM存储位置 在VMware上安装Linux(发行版CentOS7)安装包获取CentOS7 安装 Mac系…

LeetCode热题100--19.删除链表的倒数第N个结点--中等

1. 题目 给你一个链表&#xff0c;删除链表的倒数第 n 个结点&#xff0c;并且返回链表的头结点。 示例 1&#xff1a; 输入&#xff1a;head [1,2,3,4,5], n 2 输出&#xff1a;[1,2,3,5] 示例 2&#xff1a; 输入&#xff1a;head [1], n 1 输出&#xff1a;[] 示例…

git学习与使用(远程仓库、分支、工作流)

文章目录 前言简介git的工作流程git的安装配置git环境&#xff1a;git config --globalgit的基本使用新建目录初始化仓库&#xff08;repository&#xff09;添加到暂存区新增/修改/删除 文件状态会改变 提交到仓库查看提交&#xff08;commit&#xff09;的历史记录git其他命令…

《Android 应用开发基础教程》——第十四章:Android 多线程编程与异步任务机制(Handler、AsyncTask、线程池等)

目录 第十四章&#xff1a;Android 多线程编程与异步任务机制&#xff08;Handler、AsyncTask、线程池等&#xff09; &#x1f538; 14.1 为什么需要多线程&#xff1f; &#x1f538; 14.2 Handler Thread 模型 ✦ 使用 Handler 与 Thread 进行线程通信 ✦ 简要说明&am…

【JVM 01-引言入门篇】

JVM 引言篇01 笔记记录 1. 什么是JVM&#xff1f;2. 学习JVM有什么用&#xff1f;3. 常见的JVM4. 学习路线 学习资料来源-b站黑马 1. 什么是JVM&#xff1f; 定义&#xff1a;Java虚拟机&#xff08;Java Virtual Machine 简称JVM&#xff09;是运行所有Java程序的抽象计算机&a…

Pandas数据规整

&#xff08;1&#xff09;层次化索引 1.创建带层次化索引的df 第一种&#xff0c;直接创建 import pandas as pd import numpy as npdata pd.Series(np.random.randn(9),index [[a, a, a, b, b, c, c, d, d],[1, 2, 3, 1, 3, 1, 2, 2, 3]]) print(data) # a 1 -0.6416…

ThreadLocal线程本地变量在dubbo服务使用时候遇到的一个坑

我昨天遇到一个问题&#xff0c;就是我springboot项目里面有一个提供代办服务审核的dubbo接口&#xff0c;这个接口给房源项目调用&#xff0c;但是碰到一个问题就是&#xff0c;房源项目每天凌晨5点会查询满足条件过期的数据&#xff0c;然后调用我这边的代办审核dubbo接口&am…