MyBatis 动态 SQL 详解:灵活构建强大查询
MyBatis 的动态 SQL 功能是其最强大的特性之一,它允许开发者根据不同条件动态生成 SQL 语句,极大地提高了 SQL 的灵活性和复用性。本文将深入探讨 MyBatis 的动态 SQL 功能,包括 OGNL 表达式的使用以及各种动态 SQL 元素(如 if、choose、when、foreach 等)的应用场景和示例。
1.动态 SQL 概述
动态 SQL 是 MyBatis 的核心特性之一,它允许在 XML 映射文件或注解中定义灵活的 SQL 语句,根据运行时条件动态生成最终执行的 SQL。常见的应用场景包括:
- 根据不同条件构建 WHERE 子句
- 动态插入或更新字段
- 处理集合参数,实现批量操作
- 构建复杂的查询条件组合
动态 SQL 的核心是通过 OGNL(对象图导航语言)表达式来评估条件,并结合各种动态元素来生成 SQL。
2.OGNL 表达式基础
OGNL(Object Graph Navigation Language)是一种强大的表达式语言,MyBatis 使用它来解析动态 SQL 中的条件表达式。在 MyBatis 中,OGNL 表达式主要用于:
- 访问对象属性:user.username
- 调用方法:list.size()
- 执行逻辑运算:age > 18 && gender == 'M'
- 判断集合是否包含元素:list.contains('value')
示例:
<if test="username != null and username != ''">
AND username = #{username}
</if>
这里的 test 属性值就是一个 OGNL 表达式,用于判断 username 是否不为空。
3.动态 SQL 元素详解
3.1 <if> 元素
<if> 元素是最基本的动态 SQL 元素,用于条件判断。
示例:
<select id="findUser" parameterType="map" resultType="User">
SELECT * FROM users
WHERE 1=1<if test="username != null and username != ''">
AND username = #{username}</if><if test="age != null and age > 0">
AND age > #{age}</if>
</select>
这个查询会根据传入的参数动态添加条件。如果 username 不为空,则添加 username 条件;如果 age 不为空且大于 0,则添加 age 条件。
3.2 <choose>、<when>、<otherwise> 元素
<choose> 元素类似于 Java 中的 switch 语句,用于多条件选择。
示例:
<select id="findUser" parameterType="map" resultType="User">
SELECT * FROM users
WHERE 1=1<choose><when test="username != null and username != ''">
AND username = #{username}</when><when test="email != null and email != ''">
AND email = #{email}</when><otherwise>
AND age > 18</otherwise></choose>
</select>
这个查询会依次检查条件,一旦某个 <when> 条件满足,就会使用对应的 SQL 片段,其他条件将被忽略。如果所有 <when> 条件都不满足,则使用 <otherwise> 中的 SQL 片段。
3.3 <where> 元素
<where> 元素用于简化 SQL 语句中的 WHERE 子句,它会自动处理 AND 和 OR 前缀。
示例:
<select id="findUser" parameterType="map" resultType="User">
SELECT * FROM users<where><if test="username != null and username != ''">
username = #{username}</if><if test="age != null and age > 0">
AND age > #{age}</if></where>
</select>
如果第一个条件成立,<where> 元素会自动添加 WHERE 关键字;如果后面的条件以 AND 或 OR 开头,<where> 元素会自动去除这些前缀,避免 SQL 语法错误。
3.4 <set> 元素
<set> 元素用于动态更新语句,它会自动处理逗号。
示例:
<update id="updateUser" parameterType="User">
UPDATE users<set><if test="username != null and username != ''">
username = #{username},</if><if test="email != null and email != ''">
email = #{email},</if><if test="age != null">
age = #{age}</if></set>
WHERE id = #{id}
</update>
<set> 元素会自动添加 SET 关键字,并去除最后一个条件后的逗号,确保 SQL 语法正确。
3.5 <foreach> 元素
<foreach> 元素用于遍历集合,常用于 IN 条件或批量操作。
属性说明:
- collection:要遍历的集合,如 List、Array 或 Map。
- item:集合中的元素。
- index:索引,对于 List 和 Array 是位置索引,对于 Map 是键。
- open:开始符号,如 (。
- close:结束符号,如 )。
- separator:分隔符,如 ,。
示例 1:IN 条件
<select id="findUsersByIds" parameterType="list" resultType="User">
SELECT * FROM users
WHERE id IN<foreach item="id" collection="list" open="(" separator="," close=")">
#{id}</foreach>
</select>
示例 2:批量插入
<insert id="insertUsers" parameterType="list">
INSERT INTO users (username, email, age)
VALUES<foreach item="user" collection="list" separator=",">
(#{user.username}, #{user.email}, #{user.age})</foreach>
</insert>
3.6 <trim> 元素
<trim> 元素是一个通用的格式化元素,可以用来定制 <where> 和 <set> 元素的功能。
属性说明:
- prefix:添加前缀。
- prefixOverrides:去除前缀。
- suffix:添加后缀。
- suffixOverrides:去除后缀。
替代 <where> 元素:
<trim prefix="WHERE" prefixOverrides="AND |OR ">
...
</trim>
替代 <set> 元素:
<trim prefix="SET" suffixOverrides=",">
...
</trim>
3.7 <sql> 和 <include> 元素
<sql> 元素用于定义可重用的 SQL 片段,<include> 元素用于引用这些片段。
示例:
<sql id="userColumns">
id, username, email, age
</sql><select id="findUser" parameterType="int" resultType="User">
SELECT <include refid="userColumns"/>
FROM users
WHERE id = #{id}
</select>
4.动态 SQL 工作流程
下面是一个动态 SQL 执行的流程图,展示了 MyBatis 如何处理动态 SQL:
SQL 执行请求
|
v
获取映射文件中的 SQL 模板
|
v
解析动态 SQL 元素和 OGNL 表达式
|
v
根据条件生成最终 SQL 语句
|
v
参数处理和类型转换
|
v
执行最终生成的 SQL 语句
|
v
返回结果
5.综合示例
下面是一个综合示例,展示如何使用多种动态 SQL 元素构建复杂查询:
<mapper namespace="com.example.mapper.UserMapper"><!-- 定义可重用的列 --><sql id="userColumns">
id, username, email, age, gender</sql><!-- 复杂查询示例 --><select id="searchUsers" parameterType="map" resultType="User">
SELECT <include refid="userColumns"/>
FROM users<where><choose><when test="keyword != null and keyword != ''">
(username LIKE CONCAT('%', #{keyword}, '%')
OR email LIKE CONCAT('%', #{keyword}, '%'))</when><otherwise>
1=1</otherwise></choose><if test="ageRange != null and ageRange.size() == 2">
AND age BETWEEN #{ageRange[0]} AND #{ageRange[1]}</if><if test="genders != null and genders.size() > 0">
AND gender IN<foreach item="gender" collection="genders" open="(" separator="," close=")">
#{gender}</foreach></if></where><choose><when test="sortField != null and sortField != ''">
ORDER BY ${sortField}<if test="sortOrder != null and sortOrder != ''">
${sortOrder}</if></when><otherwise>
ORDER BY id DESC</otherwise></choose><if test="offset != null and limit != null">
LIMIT #{offset}, #{limit}</if></select><!-- 动态更新示例 --><update id="updateUser" parameterType="User">
UPDATE users<set><if test="username != null and username != ''">
username = #{username},</if><if test="email != null and email != ''">
email = #{email},</if><if test="age != null">
age = #{age},</if><if test="gender != null and gender != ''">
gender = #{gender}</if></set>
WHERE id = #{id}</update><!-- 批量插入示例 --><insert id="batchInsert" parameterType="list">
INSERT INTO users (username, email, age, gender)
VALUES<foreach item="user" collection="list" separator=",">
(#{user.username}, #{user.email}, #{user.age}, #{user.gender})</foreach></insert>
</mapper>
6.动态 SQL 最佳实践
1. 保持表达式简洁:避免在 OGNL 表达式中编写复杂的逻辑,保持表达式简单易懂。
2. 合理使用 <where> 和 <set>:它们可以自动处理 SQL 语法问题,减少错误。
3. 使用 <sql> 和 <include> 提高复用性:将常用的 SQL 片段提取出来,便于维护。
4. 谨慎使用 ${}:${} 会直接替换参数,存在 SQL 注入风险,应尽量使用 #{}。
5. 避免过度复杂的动态 SQL:如果动态 SQL 过于复杂,考虑拆分成多个简单的 SQL 语句。
6. 测试动态 SQL:由于动态 SQL 的灵活性,建议编写单元测试确保各种条件下生成的 SQL 正确。
7.总结
MyBatis 的动态 SQL 功能通过 OGNL 表达式和各种动态元素,为开发者提供了强大而灵活的 SQL 构建能力。无论是简单的条件查询,还是复杂的批量操作,动态 SQL 都能轻松应对。通过合理使用动态 SQL,可以提高 SQL 的复用性和可维护性,减少重复代码,使数据库操作更加高效。
在实际开发中,需要根据业务需求选择合适的动态 SQL 元素,遵循最佳实践,避免陷入过度复杂的动态 SQL 陷阱。掌握动态 SQL 的使用,是成为一名高效的 MyBatis 开发者的关键一步。
相关文章:
MyBatis 动态 SQL 详解:灵活构建强大查询
MyBatis 的动态 SQL 功能是其最强大的特性之一,它允许开发者根据不同条件动态生成 SQL 语句,极大地提高了 SQL 的灵活性和复用性。本文将深入探讨 MyBatis 的动态 SQL 功能,包括 OGNL 表达式的使用以及各种动态 SQL 元素(如 if、c…...
从 “金屋藏娇” 到 自然语言处理(NLP)
文章目录 从两个问题理解自然语言处理(NLP)1、汉武帝喜欢阿娇吗1. 政治联姻的背景2. 早期情感与后期疏远3. 历史评价的复杂性4. 现代视角结论 2、刘彻和淮南王关系一、背景:诸侯王与中央的矛盾二、刘彻与刘安的互动三、深层原因与历史评价结论…...

vue3 ElMessage提示语换行渲染
在 ElMessage.error 中使用换行符 \n 并不会实现换行,因为 ElMessage 默认会将字符串中的换行符忽略。要实现换行显示,可以使用 HTML 标签 <br> 并结合 ElMessage 的 dangerouslyUseHTMLString 选项。以下是实现换行提示的代码示例: i…...

Java 微服务架构设计:服务拆分与服务发现的策略
Java 微服务架构设计:服务拆分与服务发现的策略 微服务架构作为一种热门的软件架构风格,在 Java 领域有着广泛的应用。它通过将系统拆分为一组小型服务来实现更灵活、可扩展的系统设计。在微服务架构中,服务拆分和服务发现是两个关键环节。本…...

华为OD机试真题——二叉树中序遍历(2025A卷:200分)Java/python/JavaScript/C++/C语言/GO六种最佳实现
2025 A卷 200分 题型 本文涵盖详细的问题分析、解题思路、代码实现、代码详解、测试用例以及综合分析; 并提供Java、python、JavaScript、C++、C语言、GO六种语言的最佳实现方式! 2025华为OD真题目录+全流程解析/备考攻略/经验分享 华为OD机试真题《二叉树中序遍历》: 目录 …...
解决 Go 中 `loadinternal: cannot find runtime/cgo` 错误
在 Go 开发中,loadinternal: cannot find runtime/cgo 是一个相对不常见但可能令人困惑的错误。这个错误通常与 CGO 的使用和配置有关。本文将探讨这个错误的成因,并提供解决方案,帮助你在未来的开发中避免类似问题。 错误背景 在 Go 项目中…...

VSCode + GD32F407 构建烧录
前言 最近调试一块 GD32F407VET6(168Mhz,8Mhz晶振) 板子时,踩了一些“启动失败”的坑。本以为是时钟配置有误,最后发现是链接脚本(.ld 文件)没有配置好,导致程序根本没能正常执行 ma…...

Linux研学-入门命令
一 目录介绍 1 介绍 Linux与Windows在目录结构组织上差异显著:Linux采用树型目录结构,以单一根目录/为起点,所有文件和子目录由此向下延伸形成层级体系,功能明确的目录各司其职,使文件系统层次清晰、逻辑连贯…...
Hive在实际应用中,如何选择合适的JOIN优化策略?
在实际应用中选择Hive JOIN优化策略时,需综合考虑数据规模、分布特征、表结构设计、集群资源及业务需求。以下是具体的决策流程和参考标准: 一、数据特征分析 1. 统计数据规模 通过DESCRIBE FORMATTED table_name查看表大小和分区信息。使用SELECT CO…...

设计模式之结构型:桥接模式
桥接模式(Bridge Pattern) 定义 桥接模式是一种结构型设计模式,通过将抽象部分与实现部分分离,使它们可以独立变化。它通过组合代替继承,解决多层继承导致的类爆炸问题,适用于多维度变化的场景(如形状与颜…...

监控 Oracle Cloud 负载均衡器:使用 Applications Manager 释放最佳性能
设想你正在运营一个受欢迎的在线学习平台,在考试前的高峰期,平台流量激增。全球的学生同时登录,观看视频、提交作业和参加测试。如果 Oracle Cloud 负载均衡器不能高效地分配流量,或者后端服务器难以应对负载,学生可能…...

早发现=早安心!超导心磁图如何捕捉早期病变信号?
随着生活节奏的加快,心血管疾病已成为威胁人们健康的“隐形杀手”。据国家心血管病中心发布的《中国心血管健康与疾病报告2022》显示,我国心血管病现患者人数已高达3.3亿,每5例死亡中就有2例死于心血管病。这一数据触目惊心,提醒我…...

使用Vditor将Markdown文档渲染成网页(Vite+JS+Vditor)
1. 引言 编写Markdown文档现在可以说是程序员的必备技能了,因为Markdown很好地实现了内容与排版分离,可以让程序员更专注于内容的创作。现在很多技术文档,博客发布甚至AI文字输出的内容都是以Markdown格式的形式输出的。那么,Mar…...
Python打卡DAY40
知识点回顾: 彩色和灰度图片测试和训练的规范写法:封装在函数中展平操作:除第一个维度batchsize外全部展平dropout操作:训练阶段随机丢弃神经元,测试阶段eval模式关闭dropout 作业:仔细学习下测试和训练代码…...

OPC Client第6讲(wxwidgets):Logger.h日志记录文件(单例模式);登录后的主界面
接上一讲三、2、2>4》,创建logger.h和helper_t.h里的gettime函数 即解决下图的报红 同时,接上一讲二、3、点击“确认”按钮后,进入MainFrame.h对应的下述界面,此讲下图进行实现 一、创建Logger.h:日志记录文件&…...
CesiumInstancedMesh 实例
CesiumInstancedMesh 实例 import * as Cesium from cesium;// Three.js 风格的 InstancedMesh 类, https://threejs.org/docs/#api/en/objects/InstancedMesh export class CesiumInstancedMesh {/*** Creates an instance of InstancedMesh.** param {Cesium.Geometry} geom…...

单细胞注释前沿:CASSIA——无参考、可解释、自动化细胞注释的大语言模型
细胞类型注释是单细胞RNA-seq分析的重要步骤,目前有许多注释方法。大多数注释方法都需要计算和特定领域专业知识的结合,而且经常产生不一致的结果,难以解释。大语言模型有可能在减少人工输入和提高准确性的同时扩大可访问性,但现有…...

历年武汉大学计算机保研上机真题
2025武汉大学计算机保研上机真题 2024武汉大学计算机保研上机真题 2023武汉大学计算机保研上机真题 在线测评链接:https://pgcode.cn/school 分段函数计算 题目描述 写程序计算如下分段函数: 当 x > 0 x > 0 x>0 时, f ( x ) …...
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(30):みます
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(30):みます 1、前言(1)情况说明(2)工程师的信仰2、知识点(1)ように 復習:1、ように Change12、ように Ideal state(理想(りそう)の状態(じょうたい))3、V辞書・Vない ようにしています いつも気をつけて…...

AR-HUD 光波导方案优化难题待解?OAS 光学软件来破局
波导-HUD系统案例分析 简介 光波导技术凭借其平板超薄结构和强大的二维扩展能力,在解决AR-HUD问题方面展现出显著优势。一方面,其独特的结构特性能够大幅减小对光机体积的需求,成为 HUD 未来发展的重要技术方向;另一方面…...

火狐安装自动录制表单教程——仙盟自动化运营大衍灵机——仙盟创梦IDE
打开火狐插件页面 安装完成 使用 功能 录制浏览器操作 录入地址 开始操作 录制完成 在当今快速发展的软件开发生态中,自动化测试已从一种新兴技术手段,转变为保障软件质量与开发效率不可或缺的关键环节。其重要性体现在多个维度,同时&#x…...

线程池的详细知识(含有工厂模式)
前言 下午学习了线程池的知识。重点探究了ThreadPoolExecutor里面的各种参数的含义。我详细了解了这部分的知识。其中有一个参数涉及工厂模式,我将这一部分知识分享给大家~ 线程池的详细介绍(含工厂模式) 结语 分享到此结束啦。byebye~...

木愚科技闪亮第63届高博会 全栈式智能教育解决方案助力教学升级
5月23日,第63届高等教育博览会在长春东北亚国际博览中心开幕,木愚科技积极筹备,奔赴展会现场。彼时,木愚科技企业领导及相关职能部门负责人亲临展位指导工作,通过特装展位、资料发放及现场交流等方式,全方位…...

Proteus寻找元器件(常见)
一 元件库 二 找元件 1 主控 32 51 输入 stm32 AT89c51 2 找屏幕 oled 3 找按键button 4 电阻、电容 res cap 5 电机驱动 l298n 6 电机 motor 7 滑动变阻器 pot 8 找电源和 GND 9 找晶振 选择 D 开头的 CRYSTAL 10 网络标签...

RK3566 Android12 HG24C02MM/TR EEPROM适配
一、背景 近期项目中,有一个需求,要使用RK3566 Android12平台适配一款HG24C02MM/TR EEPROM芯片,通过i2c实现主板与EEPROM芯片的数据通讯。废话不多说,来看资料。 二、芯片资料 HG24C02 / HG24C04 / HG24C08 / HG24C16是提供2048…...

IoTDB 集成 DBeaver,简易操作实现时序数据清晰管理
数据结构一目了然,跨库分析轻松实现,方便 IoTDB “内部构造”管理! 随着物联网场景对时序数据处理需求激增,时序数据库与数据库管理工具的集成尤为关键。作为数据资产的 “智能管家”,借助数据库管理工具的可视化操作界…...

sqli-labs第二十八关——Trick with ‘union select‘
一:分析 这一关的提示和上一关一样,所以我们查看源码,屏蔽了注释符,空格,union,select等关键词 分析这一条源码的几个新增添符号 \s: 匹配任何的空白字符(普通空格,\t&…...

mapbox高阶,PMTiles介绍,MBTiles、PMTiles对比,加载PMTiles文件
👨⚕️ 主页: gis分享者 👨⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨⚕️ 收录于专栏:mapbox 从入门到精通 文章目录 一、🍀前言1.1 ☘️mapboxgl.Map 地图对象1.2 ☘️mapboxgl.Map style属性1.3 ☘️Fill面图层样式1.4 ☘️PMTiles介绍1.5…...
Go语言通道如何实现通信
在Go语言中,通道(channel)是一种内置的数据结构,用于在不同的goroutine之间进行通信和同步。通道提供了一种安全且有效的方式来传递数据,避免了数据竞争和死锁等问题。 要在Go语言中使用通道进行通信,你需…...

投稿 IEEE Transactions on Knowledge and Data Engineering 注意事项
投稿 IEEE Transactions on Knowledge and Data Engineering 注意事项 要IEEE overleaf 模板私信,我直接给我自己论文,便于编辑 已经投稿完成了,有一些小坑 准备工作 注册IEEE账户:若没有IEEE账户,需前往IEEE官网注册。注册成功后,可用于登录投稿系统。现在新的系统,…...