MyBatis-Plus 中 nested() 与 and() 方法详解(最佳实践场景)

2025-07-03 17:50

本文主要是介绍MyBatis-Plus 中 nested() 与 and() 方法详解(最佳实践场景),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《MyBatis-Plus中nested()与and()方法详解(最佳实践场景)》在MyBatis-Plus的条件构造器中,nested()和and()都是用于构建复杂查询条件的关键方法,但...

MyBatis-Plus 中nested()与and()方法详解

在 MyBatis-Plus 的条件构造器中,nested()and() 都是用于构建复杂查询条件的关键方法,但它们在功能和使用场景上有显著差异。

一、核心区别对比

特性and()nested()
主要用途连接多个条件(默认AND)创建嵌套查询条件
括号生成❌ 不生成括号✅ 自动生成括号包裹内部条件
逻辑关系与外部条件默认AND连接可作为独立逻辑单元
复杂查询支持有限,适合简单条件组合强大,支持任意层级嵌套
OR 条件处理需要显式使用or()天然支持OR条件嵌套
SQL 表现平铺条件生成带括号的子表达式

二、方法详解

1.and()方法

  • 功能:连接多个查询条件,默认使用 AND 逻辑
  • 特点
    • 不会生成括号
    • 适用于简单的条件组合
    • 当需要 OR 逻辑时需显式调用 or()
使用示例:
// 查询年龄大于18且姓张的用户
QueryWrapper<User> wrapper = new QueryWrapper<>();
wrapper.gt("age", 18)
       .and(wq -> wq.like("name", "张"));

生成 SQL

WHERE age > 18 AND name LIKE '%张%'

2.nested()方法

  • 功能:创建嵌套查询条件(自动添加括号)
  • 特点
    • 自动生成括号包裹内部条件
    • 支持任意层级的复杂嵌套
    • 特别适合处理 OR 逻辑组合
    • 使SQL逻辑更清晰
使用示例:
// 查询(年龄>30 或 工资>10000) 且 状态为激活的用户
QueryWrapper<User> wrapper = new QueryWrapper<>();
wrapper.nested(nq -> 
        nq.gt("age", 30).or().gt("salary", 10000))
    .eq("status", 1);

生成 SQL

WHERE (age > 30 OR salary > 10000) AND status = 1

三、高级用法对比

1. 复杂 OR 条件处理

使用 and() 实现:

wrapper.and(wq -> wq.gt("age", 30).or().gt("salary", 10China编程000))
       .eq("status", 1);
-- 逻辑错误!实际生成:
WHERE (age > 30 OR salary > 10000 AND status = 1)
-- 正确应使用括号包裹OR条件

使用 nested() 实现:

wrapper.nested(nq -> nq.gt("age", 30).or().gt("salary", 10000))
       .eq("status", 1);
-- 正确生成:
WHERE (age > 30 OR salary > 10000) AND status = 1

2. 多层级嵌套

nested() 支持无限嵌套:

wrapper.nested(nq1 -> 
    nq1.eq("type", 1)
       .nested(nq2 -> 
           nq2.lt("age", 20).or().gt("age", 60))
    )
    .like("name", "张");

生成 SQL

WHERE (type = 1 AND (age < 20 OR age > 60)) 
AND name LIKE '%张%'

四、最佳实践场景

使用and()当:

  1. 构建简单的 AND 条件组合
  2. 不需要改变条件优先级
  3. 查询条件逻辑简单平铺

使用nested()当:

  1. 需要处理 OR 条件组合
  2. 需要改变条件运算优先级
  3. 构建复杂嵌套查询
  4. 需要清晰分隔逻辑单元
  5. 处理多条件分支逻辑

混合使用示例:

wrapper.eq("dept_id", 10)
       .nested(nq -> 
           nq.like("name", "张")
             .or()
             .nested(nq2 -> 
                 nq2.gt("salary", 10000).lt("age", 40))
       );

生成 SQL

WHERE dept_id = 10 
AND (name LIKE '%张%' OR (salary > 10000 AND age < 40))

五、原理分析

1.and()实现原理

public Children and(Consumer<Param> consumer) {
    return addNestedCondition(true, consumer);
}
  • 内部调用 addNestedCondition
  • 参数 true 表示使用 AND 连接
  • 不添加括号

2.nested()实现原理

public Children nested(Consumer<Param> consumer) {
    return addNestedCondition(false, consumer);
}
  • 参数 faChina编程lse 表示不指定连接类型
  • 自动添加括号包裹表达式
  • 可作为独立逻辑单元嵌入

六、常见误区

  1. 错误认为 and() 会加括号

    • ✅ 只有 nested() 会生成括号
  2. 在 OR 条件中错误使用 and()

    // 错误:OR条件未正确包裹
    wrapper.eq("a", 1)
           .or()
           .and(wq -> wq.eq("b", 2).eq("c", 3));
    // 正确:使用nested
    wrapper.eq("a", 1)
           .or()
           .nested(nq -> nq.eq("b", 2).eq("c", 3));
  3. 忽略条件优先级

    // 错误:AND/OR优先级问题
    wrapper.eq("status", 1)
           .or()
           .eq("deleted", 0)
           .eq("name", "test");
    // 正确:使用nested明确优先级
    wrapper.nested(nq -> nq.eq("status", 1).or().eq("deleted", 0))
           .eq("name", "test");

七、性能考虑

  1. 数据库优化

    • 嵌套条件可能影响查询优化器的工作
    • 避免超过3层嵌套(影响可读性和性能)
  2. 索引利用

    // 好的实践:索引列前置
    wrapper.nested(nq -> 
        nq.eq("indexed_column1", val1)  // 索引列
           .or()
           .eq("indexed_column2", val2) // 索引列
    );
    // 避免:非索引列前置
    wrapper.nested(nq -> 
        nq.eq("non_indexed", val)      // 非索引列
           .or()
           .eq("indexed_column", val)   // 索引列
    );

总结

场景推荐方法
简单AND条件组合and()
OR条件组合nested()
改变条件优先级nested()
多层级复杂查询nested()
逻辑单元清晰分隔nested()lgCKoXa

经验法则

当需要括号来明确查询逻辑时,总是使用 nested()
当只是简单添加lgCKoXaAND条件时,使用 and()

正确使用这两个方法可以使你的MyBatis-Plus查询:

  1. 逻辑更清晰
  2. 优先级更明确
  3. 更易维护
  4. 充分利用数据库优化能力
  5. 避免难以调试的逻辑错误

到此这篇关于MyBatis-Plus 中 nested() 与 and() 方法详解的文章就介绍到这了,更多相关MyBatis-Plus nested() 与 and() 内容请搜索编程China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持China编程(www.cppcnphps.com)!

这篇关于MyBatis-Plus 中 nested() 与 and() 方法详解(最佳实践场景)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java利用@SneakyThrows注解提升异常处理效率详解

《Java利用@SneakyThrows注解提升异常处理效率详解》这篇文章将深度剖析@SneakyThrows的原理,用法,适用场景以及隐藏的陷阱,看看它如何让Java异常处理效率飙升50%,感兴趣的... 目录前言一、检查型异常的“诅咒”:为什么Java开发者讨厌它1.1 检查型异常的痛点1.2 为什么说

MySQL的配置文件详解及实例代码

《MySQL的配置文件详解及实例代码》MySQL的配置文件是服务器运行的重要组成部分,用于设置服务器操作的各种参数,下面:本文主要介绍MySQL配置文件的相关资料,文中通过代码介绍的非常详细,需要... 目录前言一、配置文件结构1.[mysqld]2.[client]3.[mysql]4.[mysqldum

使用Java读取本地文件并转换为MultipartFile对象的方法

《使用Java读取本地文件并转换为MultipartFile对象的方法》在许多JavaWeb应用中,我们经常会遇到将本地文件上传至服务器或其他系统的需求,在这种场景下,MultipartFile对象非... 目录1. 基本需求2. 自定义 MultipartFile 类3. 实现代码4. 代码解析5. 自定

Python文本相似度计算的方法大全

《Python文本相似度计算的方法大全》文本相似度是指两个文本在内容、结构或语义上的相近程度,通常用0到1之间的数值表示,0表示完全不同,1表示完全相同,本文将深入解析多种文本相似度计算方法,帮助您选... 目录前言什么是文本相似度?1. Levenshtein 距离(编辑距离)核心公式实现示例2. Jac

springboot2.1.3 hystrix集成及hystrix-dashboard监控详解

《springboot2.1.3hystrix集成及hystrix-dashboard监控详解》Hystrix是Netflix开源的微服务容错工具,通过线程池隔离和熔断机制防止服务崩溃,支持降级、监... 目录Hystrix是Netflix开源技术www.chinasem.cn栈中的又一员猛将Hystrix熔

Java调用Python脚本实现HelloWorld的示例详解

《Java调用Python脚本实现HelloWorld的示例详解》作为程序员,我们经常会遇到需要在Java项目中调用Python脚本的场景,下面我们来看看如何从基础到进阶,一步步实现Java与Pyth... 目录一、环境准备二、基础调用:使用 Runtime.exec()2.1 实现步骤2.2 代码解析三、

C#高效实现Word文档内容查找与替换的6种方法

《C#高效实现Word文档内容查找与替换的6种方法》在日常文档处理工作中,尤其是面对大型Word文档时,手动查找、替换文本往往既耗时又容易出错,本文整理了C#查找与替换Word内容的6种方法,大家可以... 目录环境准备方法一:查找文本并替换为新文本方法二:使用正则表达式查找并替换文本方法三:将文本替换为图

SQL Server 查询数据库及数据文件大小的方法

《SQLServer查询数据库及数据文件大小的方法》文章介绍了查询数据库大小的SQL方法及存储过程实现,涵盖当前数据库、所有数据库的总大小及文件明细,本文结合实例代码给大家介绍的非常详细,感兴趣的... 目录1. 直接使用SQL1.1 查询当前数据库大小1.2 查询所有数据库的大小1.3 查询每个数据库的详

python之uv使用详解

《python之uv使用详解》文章介绍uv在Ubuntu上用于Python项目管理,涵盖安装、初始化、依赖管理、运行调试及Docker应用,强调CI中使用--locked确保依赖一致性... 目录安装与更新standalonepip 安装创建php以及初始化项目依赖管理uv run直接在命令行运行pytho

Springboot项目构建时各种依赖详细介绍与依赖关系说明详解

《Springboot项目构建时各种依赖详细介绍与依赖关系说明详解》SpringBoot通过spring-boot-dependencies统一依赖版本管理,spring-boot-starter-w... 目录一、spring-boot-dependencies1.简介2. 内容概览3.核心内容结构4.