news 2026/6/9 19:48:13

Mybatis-Plus中QueryWrapper 与 LambdaQueryWrapper 的区别

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Mybatis-Plus中QueryWrapper 与 LambdaQueryWrapper 的区别

QueryWrapper 与 LambdaQueryWrapper 的区别

1.基本区别

QueryWrapper

java

// 使用字符串表示字段名 QueryWrapper<User> queryWrapper = new QueryWrapper<>(); queryWrapper.eq("name", "张三") .gt("age", 18) .like("email", "@qq.com");

LambdaQueryWrapper

java

// 使用Lambda表达式引用字段 LambdaQueryWrapper<User> lambdaWrapper = new LambdaQueryWrapper<>(); lambdaWrapper.eq(User::getName, "张三") .gt(User::getAge, 18) .like(User::getEmail, "@qq.com");

2.核心区别对比

特性QueryWrapperLambdaQueryWrapper
字段引用方式字符串硬编码Lambda表达式
类型安全性编译期不检查字段是否存在编译期检查,类型安全
重构友好性重命名字段需手动修改字符串IDE自动重构
代码可读性一般更好,直接关联实体类
性能相同相同
防SQL注入都防SQL注入都防SQL注入

3.详细对比

3.1 类型安全性

java

// QueryWrapper - 编译期不报错,运行时报错 queryWrapper.eq("nam", "张三"); // 拼写错误,编译时不检查 // LambdaQueryWrapper - 编译期报错 lambdaWrapper.eq(User::getNam, "张三"); // 编译时报错:找不到getNam方法

3.2 重构友好性

java

// QueryWrapper - 重命名需要手动修改 // 假设把字段 name 改为 username queryWrapper.eq("name", "张三"); // ❌ 需要手动改为 "username" // LambdaQueryWrapper - IDE自动重构 lambdaWrapper.eq(User::getName, "张三"); // ✅ 重命名实体类字段时自动更新

3.3 复杂查询对比

java

// QueryWrapper QueryWrapper<CapitalInfo> queryWrapper = new QueryWrapper<>(); if (StringUtils.isNotBlank(capitalName)) { queryWrapper.like("capital_name", capitalName); } if (capitalYear != null) { queryWrapper.eq("capital_year", capitalYear); } // LambdaQueryWrapper LambdaQueryWrapper<CapitalInfo> lambdaWrapper = new LambdaQueryWrapper<>(); lambdaWrapper.like(StringUtils.isNotBlank(capitalName), CapitalInfo::getCapitalName, capitalName) .eq(capitalYear != null, CapitalInfo::getCapitalYear, capitalYear);

4.实际使用示例

QueryWrapper 使用场景

java

// 1. 动态字段名(字段名是变量) String fieldName = getFieldNameFromRequest(); // 可能是 "name" 或 "username" QueryWrapper<User> wrapper = new QueryWrapper<>(); wrapper.eq(fieldName, "张三"); // Lambda方式无法实现 // 2. 复杂动态SQL拼接 QueryWrapper<User> wrapper = new QueryWrapper<>(); if (condition1) { wrapper.eq("field1", value1); } else { wrapper.isNull("field1"); } if (condition2) { wrapper.in("field2", Arrays.asList(1, 2, 3)); } // 3. 表连接查询 QueryWrapper<User> wrapper = new QueryWrapper<>(); wrapper.eq("u.status", 1) .eq("r.role_type", "admin") .apply("u.role_id = r.id");

LambdaQueryWrapper 使用场景

java

// 1. 常规查询(推荐) LambdaQueryWrapper<CapitalInfo> wrapper = new LambdaQueryWrapper<>(); wrapper.eq(CapitalInfo::getCapitalState, 1) .like(CapitalInfo::getCapitalName, "测试") .orderByDesc(CapitalInfo::getCreateTime); // 2. 链式调用 List<CapitalInfo> list = capitalInfoMapper.selectList( new LambdaQueryWrapper<CapitalInfo>() .eq(CapitalInfo::getCapitalYear, 2024) .in(CapitalInfo::getCapitalType, Arrays.asList("A", "B", "C")) .groupBy(CapitalInfo::getCapitalSource) ); // 3. 条件判断简化 LambdaQueryWrapper<CapitalInfo> wrapper = new LambdaQueryWrapper<>(); wrapper.eq(queryDTO.getCapitalState() != null, CapitalInfo::getCapitalState, queryDTO.getCapitalState()) .like(StringUtils.isNotBlank(queryDTO.getCapitalName()), CapitalInfo::getCapitalName, queryDTO.getCapitalName());

5.性能比较

两者在性能上没有区别,最终都会转换为相同的SQL语句:

sql

-- 两者生成的SQL相同 SELECT * FROM capital_info WHERE capital_state = 1 AND capital_name LIKE '%测试%' ORDER BY create_time DESC

6.最佳实践建议

推荐使用 LambdaQueryWrapper 的情况:

  1. 查询条件固定,字段名不会动态变化

  2. 注重代码可维护性和类型安全

  3. 实体类字段可能重构的情况

  4. 新项目开发,优先使用Lambda方式

推荐使用 QueryWrapper 的情况:

  1. 字段名需要动态生成

  2. 多表连接查询,涉及多个表的字段

  3. 处理遗留代码,保持一致性

  4. 构建复杂动态SQL,使用字符串更灵活

7.混合使用示例

java

// 可以使用QueryWrapper的lambda()方法 QueryWrapper<CapitalInfo> wrapper = new QueryWrapper<>(); wrapper.lambda() // 转换为Lambda方式 .eq(CapitalInfo::getCapitalState, 1) .like(CapitalInfo::getCapitalName, "测试"); // 或者反过来 LambdaQueryWrapper<CapitalInfo> lambdaWrapper = new LambdaQueryWrapper<>(); lambdaWrapper.eq(CapitalInfo::getCapitalState, 1); QueryWrapper<CapitalInfo> queryWrapper = lambdaWrapper.getWrapper(); // 可以继续添加字符串方式的查询条件 queryWrapper.apply("DATE(create_time) = CURDATE()");

8.结论

方面推荐原因
新项目开发LambdaQueryWrapper类型安全,重构友好
字段名固定LambdaQueryWrapper减少硬编码错误
字段名动态QueryWrapper灵活性强
代码可读性LambdaQueryWrapper直观明了
兼容性QueryWrapper所有场景都适用

建议:在大部分业务场景下,优先使用LambdaQueryWrapper,因为它提供了更好的开发体验和代码安全性。只有在需要动态字段名或处理复杂SQL时,才使用QueryWrapper

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/9 19:11:23

开源模型部署痛点全解析:以Image-to-Video为例

开源模型部署痛点全解析&#xff1a;以Image-to-Video为例 引言&#xff1a;从理想到现实的鸿沟 近年来&#xff0c;随着AIGC&#xff08;人工智能生成内容&#xff09;技术的爆发式发展&#xff0c;图像转视频&#xff08;Image-to-Video, I2V&#xff09; 成为多模态生成领域…

作者头像 李华
网站建设 2026/5/29 13:10:45

Sambert-HifiGan语音合成服务的性能基准测试

Sambert-HifiGan语音合成服务的性能基准测试 引言&#xff1a;中文多情感语音合成的技术演进与现实需求 随着智能客服、虚拟主播、有声阅读等应用场景的不断扩展&#xff0c;高质量的中文多情感语音合成&#xff08;Text-to-Speech, TTS&#xff09; 已成为AI交互系统的核心能…

作者头像 李华
网站建设 2026/5/29 11:44:50

Top10开源AI视频生成器测评:谁是真正的效率之王?

Top10开源AI视频生成器测评&#xff1a;谁是真正的效率之王&#xff1f; 在AIGC&#xff08;人工智能生成内容&#xff09;爆发式增长的今天&#xff0c;AI视频生成技术正以前所未有的速度重塑创意生产流程。从静态图像到动态视频的跨越&#xff0c;不再依赖复杂的后期制作&…

作者头像 李华
网站建设 2026/6/6 5:57:28

Sambert-HifiGan情感语音合成背后的深度学习原理

Sambert-HifiGan情感语音合成背后的深度学习原理 &#x1f4cc; 技术背景&#xff1a;从机械朗读到情感化语音合成的演进 早期的语音合成系统&#xff08;如基于拼接法或参数化统计模型的TTS&#xff09;虽然能实现“文字转语音”&#xff0c;但输出声音往往生硬、缺乏自然语调…

作者头像 李华
网站建设 2026/6/2 12:23:53

用Sambert-HifiGan为智能门锁添加语音交互功能

用Sambert-HifiGan为智能门锁添加语音交互功能 &#x1f4cc; 引言&#xff1a;让智能门锁“会说话”的技术路径 随着智能家居生态的不断演进&#xff0c;用户对设备交互体验的要求已从“能用”升级到“好用、贴心”。传统智能门锁多依赖LED提示音或手机App通知进行状态反馈&am…

作者头像 李华
网站建设 2026/5/27 10:34:25

如何用Sambert-HifiGan制作语音版菜谱?

如何用Sambert-HifiGan制作语音版菜谱&#xff1f; 引言&#xff1a;让菜谱“开口说话”——中文多情感语音合成的实用场景 在智能家居、无障碍阅读和内容创作等场景中&#xff0c;将静态文本转化为自然流畅的语音正变得越来越重要。尤其对于中老年用户或视障人群&#xff0c;语…

作者头像 李华