MyBatis动态SQL技巧与使用方法详解
本篇文章向大家介绍《MyBatis动态SQL编写技巧与方法详解》,主要包括,具有一定的参考价值,需要的朋友可以参考一下。
MyBatis动态SQL通过一系列XML标签实现灵活的SQL拼接,提升了代码可维护性和数据操作适应性。1.if标签用于条件判断,根据test属性决定是否包含SQL片段;2.where标签自动处理WHERE关键字及AND/OR前缀,避免冗余;3.set标签用于UPDATE语句,自动添加SET并处理末尾逗号;4.choose/when/otherwise实现多选一逻辑,适用于互斥条件查询;5.foreach遍历集合,常用于IN子句或批量操作;6.trim提供灵活控制,自定义前后缀及覆盖内容。处理空值时,通常在test中同时判断null和空字符串,或将空字符串转为NULL。复杂组合查询可通过嵌套if、手动加括号等方式构建逻辑。批量操作主要依赖foreach标签,支持INSERT、UPDATE、DELETE。使用#{}防止SQL注入,${}仅限动态表名列名。建议服务层预处理参数,避免SQL过于复杂影响可维护性。
MyBatis动态SQL的核心在于灵活构建SQL语句,它允许我们根据不同条件,在运行时动态地拼接SQL片段,极大提升了数据操作的适应性和代码的可维护性。这玩意儿,用好了能让你的Mapper文件活起来,告别一堆冗余的SQL分支。

解决方案
MyBatis动态SQL的实现,主要是通过它提供的一系列XML标签来完成的。这些标签就像一个个乐高积木,你可以根据业务逻辑自由组合,拼装出最终的SQL语句。
if
标签: 这是最基础的条件判断,类似于Java里的if
语句。它会根据test
属性的布尔结果来决定是否包含内部的SQL片段。<select id="findUsers" resultType="User"> SELECT * FROM user <where> <if test="username != null and username != ''"> username = #{username} </if> <if test="age != null"> AND age > #{age} </if> </where> </select>
这里我常常会遇到一个问题,就是如果所有条件都不满足,
标签会自己处理掉开头的AND
或OR
,这很省心,但如果是在非
或
标签里用if
,就得自己注意前缀了。where
标签: 专门用于处理SQL语句中的WHERE
子句。它会自动添加WHERE
关键字,并智能地处理内部第一个条件前的AND
或OR
。如果内部没有任何条件成立,它也不会生成WHERE
关键字,非常方便。set
标签: 主要用于UPDATE
语句,它会自动添加SET
关键字,并处理内部条件末尾的逗号。<update id="updateUser" parameterType="User"> UPDATE user <set> <if test="username != null and username != ''"> username = #{username}, </if> <if test="email != null and email != ''"> email = #{email}, </if> </set> WHERE id = #{id} </update>
我个人觉得
set
标签简直是更新操作的福音,以前手写得小心翼翼地处理最后一个逗号,现在完全不用操心了。choose
/when
/otherwise
标签: 类似于Java的switch
语句,用于实现多选一的逻辑。MyBatis会从多个when
条件中选择第一个满足的执行,如果没有when
满足,则执行otherwise
中的内容。<select id="searchProducts" resultType="Product"> SELECT * FROM product <where> <choose> <when test="productName != null and productName != ''"> product_name LIKE CONCAT('%', #{productName}, '%') </when> <when test="category != null"> category = #{category} </when> <otherwise> <!-- 默认情况,比如查询所有活跃产品 --> status = 'active' </otherwise> </choose> </where> </select>
这个在需要多条件互斥查询时特别好用,比如用户只能通过商品名或分类中的一个来搜索。
foreach
标签: 强大的循环标签,用于遍历集合(List, Array, Set)或Map,常用于构建IN
子句、批量插入或批量更新等场景。<select id="findUsersByIds" resultType="User"> SELECT * FROM user WHERE id IN <foreach item="id" collection="list" open="(" separator="," close=")"> #{id} </foreach> </select>
这里的
collection
属性非常关键,如果是List,通常是list
;如果是数组,是array
;如果是Map,则是Map的键名。我有时会忘记这个,导致参数绑定失败,所以每次写都得确认下。trim
标签: 提供了最灵活的控制能力,可以自定义前缀、后缀,以及需要覆盖(删除)的前缀或后缀。当你发现where
或set
不能满足你的奇葩需求时,它就是你的救星。<trim prefix="WHERE" prefixOverrides="AND |OR " suffixOverrides=","> <if test="username != null">AND username = #{username}</if> <if test="email != null">OR email = #{email}</if> </trim>
比如我遇到过需要在
GROUP BY
后动态添加HAVING
条件,trim
就能派上用场。
在MyBatis动态SQL中处理空字符串和NULL值的最佳实践是什么?
这真是个老生常谈的问题,但又不得不提。在Java里,null
和空字符串""
是两码事,但在数据库查询里,很多时候我们希望它们行为一致,或者至少能区分开来。
最常见的处理方式是在test
属性中同时判断null
和空字符串:test="username != null and username != ''"
。这几乎成了我的肌肉记忆,每次写条件查询都会带上这个。它能有效避免因为传入空字符串而导致查询结果不符合预期,或者直接报错。
有时候,我会考虑把一些空字符串转NULL
的逻辑放到数据库层面,比如通过IFNULL
或NULLIF
函数。但这要看具体业务场景和数据库支持。另外,偶尔也会遇到参数类型不匹配导致的问题,比如数字类型字段传了空字符串。MyBatis默认的类型处理器会尝试转换,但如果转换失败就会抛异常。这时,确保前端传过来的数据类型与Mapper接口的参数类型一致,或者在业务层做预处理,会省去很多麻烦。
个人经验是,我通常倾向于在Java服务层就对传入参数进行初步校验和清洗,比如将所有业务意义上的“空”字符串统一转为null
,这样Mapper层的动态SQL可以更专注于null
判断,逻辑会更清晰。但如果业务需求就是区分空字符串和null
,那就老老实实写!= ''
和!= null
。
如何有效利用MyBatis动态SQL编写复杂的组合查询条件?
复杂的组合查询,往往意味着AND和OR的嵌套。MyBatis的动态SQL标签本身是扁平化的,但通过巧妙的组合和嵌套,完全可以构建出非常复杂的逻辑。
处理AND与OR的优先级时,默认情况下AND
的优先级高于OR
。当我们需要改变优先级时,就必须引入括号。MyBatis动态SQL本身不直接提供括号标签,但我们可以通过在SQL片段中手动添加括号来实现。
<select id="findComplexUsers" resultType="User"> SELECT * FROM user <where> <if test="conditionA != null"> condition_a = #{conditionA} </if> <if test="conditionB != null or conditionC != null"> AND ( <if test="conditionB != null"> condition_b = #{conditionB} </if> <if test="conditionC != null"> OR condition_c = #{conditionC} </if> ) </if> </where> </select>
注意看上面,AND (
和 )
是我手动加的,这是处理复杂逻辑的关键。内部的OR
条件,如果只有一个成立,MyBatis的where
标签会智能处理掉多余的OR
,但这里因为有括号,所以内部的OR
需要更小心。通常我会让内部的第一个条件不带OR
或AND
,后续的条件才带。
在某些极端复杂的场景,你甚至可以在一个大的where
标签内部,再嵌套一个trim
或另一个where
(虽然不常见,但理论上可行,不过会增加阅读难度)。我个人觉得,如果SQL复杂到需要这样嵌套,可能就要考虑是不是业务逻辑可以在服务层拆分,或者考虑使用更高级的查询工具,或者直接写存储过程了。
我的一个忠告是,动态SQL再强大,也别把它当成万能钥匙。当一个Mapper文件里的动态SQL逻辑变得像一团乱麻时,它就失去了可维护性。这时候,考虑拆分SQL,或者在Java代码层面进行一些条件判断和SQL片段的拼接(虽然这会失去MyBatis的便利,但有时是必要的妥协)。
另外,动态SQL的参数都是通过#{}
占位符绑定的,这能有效防止SQL注入,因为MyBatis会使用PreparedStatement。但如果你在动态SQL中使用了${}
,那就要特别小心了,它会直接拼接字符串,有SQL注入风险,通常只用于动态表名或列名。
MyBatis动态SQL在批量操作(插入、更新、删除)中的应用技巧有哪些?
批量操作是提升数据库性能的重要手段,MyBatis的动态SQL在这里扮演了不可或缺的角色,尤其是foreach
标签。
批量插入 (INSERT BATCH
):
这是最常见的批量操作场景。foreach
可以用来构建多条VALUES
子句。
<insert id="batchInsertUsers" parameterType="java.util.List"> INSERT INTO user (username, email, create_time) VALUES <foreach collection="list" item="user" separator=","> (#{user.username}, #{user.email}, NOW()) </foreach> </insert>
这里需要注意的是,不同的数据库对批量插入的语法支持有所不同。上述是最常见的MySQL风格。如果是Oracle,可能需要用ALL
关键字和UNION ALL
来模拟。
批量更新 (UPDATE BATCH
):
批量更新相对复杂一些,因为每个更新的条件和值可能不同。一种常见做法是利用CASE WHEN
结合foreach
。
<update id="batchUpdateStatus" parameterType="java.util.List"> UPDATE user <set> status = CASE id <foreach collection="list" item="item" index="index"> WHEN #{item.id} THEN #{item.status} </foreach> END </set> WHERE id IN <foreach collection="list" item="item" separator="," open="(" close=")"> #{item.id} </foreach> </update>
这种方式在处理少量数据时很方便,但如果批量更新的数据量非常大,比如几万条,这种CASE WHEN
的SQL可能会变得非常长,数据库执行效率会下降。这时,我通常会考虑分批次提交,或者使用数据库连接池的addBatch()
方法配合executeBatch()
来执行。
批量删除 (DELETE BATCH
):
批量删除通常也用foreach
构建IN
子句,这和批量查询的IN
子句非常相似。
<delete id="batchDeleteUsers" parameterType="java.util.List"> DELETE FROM user WHERE id IN <foreach collection="list" item="id" separator="," open="(" close=")"> #{id} </foreach>
今天关于《MyBatis动态SQL技巧与使用方法详解》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!

- 上一篇
- Java实现MinIO分片上传方法详解

- 下一篇
- Golang错误堆栈跟踪与pkg/errors使用详解
-
- 文章 · java教程 | 12分钟前 |
- RocketMQ安装配置详解教程
- 475浏览 收藏
-
- 文章 · java教程 | 19分钟前 |
- Java热修复:动态类重定义调试方法
- 111浏览 收藏
-
- 文章 · java教程 | 30分钟前 |
- Mockito模拟Futureget异常及验证方法
- 453浏览 收藏
-
- 文章 · java教程 | 33分钟前 | java 事务处理 threadlocal abstractroutingdatasource 动态数据源切换
- Java动态数据源切换配置与实现方法
- 481浏览 收藏
-
- 文章 · java教程 | 49分钟前 |
- Kafka与Java微服务整合指南
- 395浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java实战:Arthas线上诊断使用指南
- 188浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java文件统计工具:FileStats类高效实现
- 479浏览 收藏
-
- 文章 · java教程 | 2小时前 | java 天文数据 nom.tam.fits HDU FITS
- Java读取FITS天文文件方法解析
- 440浏览 收藏
-
- 文章 · java教程 | 2小时前 | SpringSecurity 资源服务器 jwt 权限控制 OAuth2
- SpringSecurityOAuth2资源服务器配置全解析
- 148浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java多格式时间戳统一处理方法
- 306浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java正则表达式高级应用技巧
- 424浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 511次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 498次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 边界AI平台
- 探索AI边界平台,领先的智能AI对话、写作与画图生成工具。高效便捷,满足多样化需求。立即体验!
- 422次使用
-
- 免费AI认证证书
- 科大讯飞AI大学堂推出免费大模型工程师认证,助力您掌握AI技能,提升职场竞争力。体系化学习,实战项目,权威认证,助您成为企业级大模型应用人才。
- 427次使用
-
- 茅茅虫AIGC检测
- 茅茅虫AIGC检测,湖南茅茅虫科技有限公司倾力打造,运用NLP技术精准识别AI生成文本,提供论文、专著等学术文本的AIGC检测服务。支持多种格式,生成可视化报告,保障您的学术诚信和内容质量。
- 563次使用
-
- 赛林匹克平台(Challympics)
- 探索赛林匹克平台Challympics,一个聚焦人工智能、算力算法、量子计算等前沿技术的赛事聚合平台。连接产学研用,助力科技创新与产业升级。
- 665次使用
-
- 笔格AIPPT
- SEO 笔格AIPPT是135编辑器推出的AI智能PPT制作平台,依托DeepSeek大模型,实现智能大纲生成、一键PPT生成、AI文字优化、图像生成等功能。免费试用,提升PPT制作效率,适用于商务演示、教育培训等多种场景。
- 576次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览