MyBatis系统学习(三)——动态SQL
MyBatis 是一款优秀的持久层框架,它通过 XML 或注解方式将 SQL 语句与 Java 对象映射起来。动态 SQL 是 MyBatis 中非常强大的功能之一,能够根据不同的条件动态生成 SQL 语句。动态 SQL 通过各种标签来灵活生成 SQL,从而避免了在代码中拼接 SQL 的复杂性和冗余性。接下来,我们会详细讲解 MyBatis 中动态 SQL 的相关知识点,涵盖动态 SQL 的元素、条件查询、更新操作及复杂查询操作。
一、MyBatis 动态 SQL 标签
动态 SQL 主要通过 MyBatis 提供的一些 XML 元素来实现,这些元素会根据传递的参数、条件等动态拼装 SQL。主要的动态 SQL 元素有:
-
<if>
标签- 用于判断条件,只有当条件成立时,才会生成相应的 SQL 片段。
- 语法:
<if test="条件">SQL 语句 </if>
- 示例:
上述 SQL 语句中,只有当<select id="findUser" parameterType="int" resultType="User">SELECT * FROM userWHERE 1=1<if test="id != null">AND id = #{id}</if> </select>
id
不为 null 时,才会拼接AND id = #{id}
这部分 SQL。
-
<choose>
,<when>
,<otherwise>
标签- 类似于 Java 中的 switch-case 语法,用于多条件选择。
<choose>
标签包含若干<when>
,如果多个<when>
条件都不满足,可以用<otherwise>
作为默认的条件。 - 语法:
<choose><when test="条件1">SQL 语句1</when><when test="条件2">SQL 语句2</when><otherwise>默认 SQL 语句</otherwise> </choose>
- 示例:
<select id="findUser" parameterType="map" resultType="User">SELECT * FROM user<where><choose><when test="name != null">name = #{name}</when><when test="age != null">age = #{age}</when><otherwise>status = 'active'</otherwise></choose></where> </select>
- 类似于 Java 中的 switch-case 语法,用于多条件选择。
-
<where>
标签- 用来自动处理 WHERE 子句前缀问题。
<where>
会智能地在第一个条件前添加WHERE
,并自动去掉多余的AND
或OR
。 - 示例:
这里,如果<select id="findUser" parameterType="map" resultType="User">SELECT * FROM user<where><if test="name != null">AND name = #{name}</if><if test="age != null">AND age = #{age}</if></where> </select>
name
和age
都为空,<where>
标签会避免生成无效的WHERE
子句。
- 用来自动处理 WHERE 子句前缀问题。
-
<set>
标签- 在更新操作中常用,类似于
<where>
,<set>
也会自动处理SET
关键字前后的逗号问题。 - 示例:
在生成 SQL 时,<update id="updateUser" parameterType="User">UPDATE user<set><if test="name != null">name = #{name},</if><if test="age != null">age = #{age},</if></set>WHERE id = #{id} </update>
<set>
标签会自动处理最后一项的逗号。
- 在更新操作中常用,类似于
-
<trim>
标签trim
标签用于自定义去掉前后多余的字符,替代where
和set
标签。它有四个重要的属性:prefix
: SQL 片段的前缀,如WHERE
或SET
。prefixOverrides
: 需要去掉的前缀,如AND
、OR
。suffix
: SQL 片段的后缀。suffixOverrides
: 需要去掉的后缀,如,
。
- 示例:
<select id="findUser" parameterType="map" resultType="User">SELECT * FROM user<trim prefix="WHERE" prefixOverrides="AND"><if test="name != null">AND name = #{name}</if><if test="age != null">AND age = #{age}</if></trim> </select>
-
<foreach>
标签- 用于处理集合类型的参数,比如
List
或数组
,通常用于IN
查询或批量插入操作。 - 重要属性:
collection
: 要遍历的集合,通常是List
或Array
。item
: 当前集合元素的别名。separator
: 元素之间的分隔符,如,
。open
和close
: 分别表示 SQL 片段的开头和结尾字符。
- 示例:
<select id="findUserByIds" parameterType="list" resultType="User">SELECT * FROM userWHERE id IN<foreach collection="list" item="id" open="(" separator="," close=")">#{id}</foreach> </select>
- 用于处理集合类型的参数,比如
二、条件查询的使用
在实际开发中,条件查询是非常常见的需求。MyBatis 中利用动态 SQL 能够灵活生成条件查询语句,通常使用 <if>
标签来控制条件的生成。
示例:简单条件查询
<select id="findUserByCondition" parameterType="User" resultType="User">SELECT * FROM user<where><if test="name != null">AND name = #{name}</if><if test="age != null">AND age = #{age}</if></where>
</select>
在上面的例子中,只有当 name
或 age
不为空时,才会生成对应的条件。这样可以避免生成无效的 SQL,提升查询效率。
示例:多条件选择查询
<select id="findUserByCondition" parameterType="User" resultType="User">SELECT * FROM user<where><choose><when test="name != null">name = #{name}</when><when test="age != null">age = #{age}</when><otherwise>status = 'active'</otherwise></choose></where>
</select>
这个例子中,choose
标签确保只有一个条件成立时才会生成对应的 SQL。如果多个条件不成立,则使用 otherwise
语句。
三、动态更新操作
动态 SQL 在更新操作中也非常有用。通常,我们在更新时会根据实际情况只更新某些字段,而不是全部字段。MyBatis 提供了 <set>
标签来方便地生成动态更新语句。
示例:动态更新用户信息
<update id="updateUser" parameterType="User">UPDATE user<set><if test="name != null">name = #{name},</if><if test="age != null">age = #{age},</if><if test="email != null">email = #{email},</if></set>WHERE id = #{id}
</update>
在这个例子中,只有不为空的字段才会被更新,<set>
标签确保了 SQL 语句中不会出现多余的逗号。
四、复杂查询操作
在复杂查询中,动态 SQL 的优势更为明显。我们可以根据多种条件动态生成查询语句,并使用 join
、子查询等进行复杂操作。
示例:动态查询用户及其订单
<select id="findUserWithOrders" resultMap="userOrderMap">SELECT u.*, o.*FROM user uLEFT JOIN orders o ON u.id = o.user_id<where><if test="u.name != null">u.name = #{u.name}</if><if test="o.status != null">AND o.status = #{o.status}</if></where>
</select>
这个查询会根据用户信息和订单状态动态生成 SQL,可以方便地查询用户及其订单。
五、总结
MyBatis 的动态 SQL 为开发者提供了灵活、可扩展的 SQL 生成方式。通过使用 <if>
、<choose>
、<where>
、<set>
、<foreach>
等标签,能够根据条件动态生成 SQL,简化了代码,避免了 SQL 拼接的复杂性和冗余性。
动态 SQL 的优势在于:
- 简化 SQL 编写:开发者可以通过 XML 或注解方式编写复杂的 SQL 逻辑。
- 提升代码可读性:通过清晰的条件控制和 SQL 生成逻辑,代码更加清晰。
- 减少代码冗余:避免了在 Java 代码中拼接 SQL 字符串的繁琐操作。
掌握动态 SQL 的使用对于 MyBatis 框架的实际应用非常重要,在复杂的业务逻辑中尤为常见。
相关文章:
MyBatis系统学习(三)——动态SQL
MyBatis 是一款优秀的持久层框架,它通过 XML 或注解方式将 SQL 语句与 Java 对象映射起来。动态 SQL 是 MyBatis 中非常强大的功能之一,能够根据不同的条件动态生成 SQL 语句。动态 SQL 通过各种标签来灵活生成 SQL,从而避免了在代码中拼接 S…...
get_property --Cmakelist之中
get_property 是 CMake 中用于获取目标、目录、变量或文件等属性的命令。它可以提取某个特定属性的值,以便在构建脚本的其他地方使用。 语法 get_property(<variable> <TYPE> <name> PROPERTY <property-name> [SET | DEFINED | BRIEF_DO…...

【Redis】Redis 典型应用 - 分布式锁原理与实现
目录 Redis 典型应⽤ - 分布式锁什么是分布式锁分布式锁的基础实现引⼊过期时间引⼊校验 id引⼊ lua引⼊ watch dog (看⻔狗)引⼊ Redlock 算法其他功能 Redis 典型应⽤ - 分布式锁 什么是分布式锁 在⼀个分布式的系统中, 也会涉及到多个节点访问同⼀个公共资源的…...
Pybind11的使用
目录 1. 引言1.1 Pybind11 简介1.2 为什么需要 Pybind11 2. 使用 Pybind11 进行 C 与 Python 交互2.1 基本用法2.2 编译与生成共享库2.2.1 在 Linux 下编译2.2.2 在 macOS 下编译2.2.3 编译选项详解 2.3 在 Python 中使用编译后的模块 3. 高级用法与注意事项3.1 绑定类和复杂数…...
鸿蒙-沉浸式pc端失效
咨询描述: 因PC北向窗口涉及沉浸式时,预计发生接口废弃导致不兼容变更,涉及接口setImmersiveModeEnabledState、setWindowLayoutFullSceen 如果应用支持沉浸式(窗口全屏且隐藏状态栏&标题栏&Dock栏)࿰…...

【资料分析】刷题日记1
第一套 第二个是相比2019年的增长率,错找为同比增长率 延申: 当出口和进口相比2019年的增长率相同时,可以用盐水解决 √ 一个假设分配(第二次是1.4取1)加法对比选项 基期倍数: 求A是B的多少倍&#x…...

nodejs+express+vue教辅课程辅助教学系统 43x2u前后端分离项目
目录 技术栈具体实现截图系统设计思路技术可行性nodejs类核心代码部分展示可行性论证研究方法解决的思路Express框架介绍源码获取/联系我 技术栈 该系统将采用B/S结构模式,开发软件有很多种可以用,本次开发用到的软件是vscode,用到的数据库是…...
96-javahashmap底层原理
HashMap是Java集合框架中的一个重要类,底层是基于哈希表实现的。哈希表是一种数据结构,可以通过哈希函数来提高查找、插入和删除操作的效率。 以下是HashMap底层实现的一些关键点: 哈希算法:HashMap使用哈希算法来计算键的哈希值…...

AI逻辑推理入门
参考数据鲸 (linklearner.com) 1. 跑通baseline 报名 申领大模型API 模型服务灵积-API-KEY管理 (aliyun.com) 跑通代码 在anaconda新建名为“LLM”的环境,并安装好相应包后,在jupyter notebook上运行baseline01.ipynb 2. 赛题解读 一般情况下,拿到一个赛题之后,我们需…...
力扣3014.输入单词需要的最少按键次数I
给你一个字符串 word,由 不同 小写英文字母组成。 电话键盘上的按键与 不同 小写英文字母集合相映射,可以通过按压按键来组成单词。例如,按键 2 对应 ["a","b","c"],我们需要按一次键来输入 "…...

【Git】远程仓库
本博客的环境是 Ubuntu/Linux 文章目录 集中式与分布式的区别远程仓库新建远程仓库克隆远程仓库向远程仓库推送从远程仓库拉取 配置Git忽略指定文件给命令配置别名 标签管理创建标签操作标签 多人协作本地分支与远程分支连接场景一场景二 集中式与分布式的区别 引荐自关于Git这…...

苹果手机铃声怎么设置自己的歌?3个方法自定义手机铃声
苹果手机内部的手机铃声库只有固定的几首铃声,且都是纯音乐,比较单调,并不是所有用户都喜欢这些铃声。那么,苹果手机铃声怎么设置自己的歌呢?小编这里有3个方法,可以教大家如何将手机铃声设置成自己喜欢的歌…...

828华为云征文|华为Flexus云服务器搭建Cloudreve私人网盘
一、华为云 Flexus X 实例:开启高效云服务新篇🌟 在云计算的广阔领域中,资源的灵活配置与卓越性能犹如璀璨星辰般闪耀。华为云 Flexus X 实例恰似一颗最为耀眼的新星,将云服务器技术推向了崭新的高度。 华为云 Flexus X 实例基于…...
【AI学习】AI绘画发展简史
无意中读了一篇发表自2022年的文章,《AI绘画何以突飞猛进? 从历史到技术突破, 一文读懂火爆的AI绘画发展史》,写的比较有意思,科普了好多我原来不知道的历史。 简单提炼一下,做个笔记。 AI绘画重要事件 2012年 Google两位大名…...
使用LangChain创建简单的语言模型应用程序【快速入门指南】
## 引言在这篇文章中,我们将展示如何使用LangChain构建一个简单的语言模型(LLM)应用程序。这个应用程序的功能是将文本从英语翻译成其他语言。尽管应用程序的逻辑相对简单,但它能够帮助我们学习如何使用LangChain进行更多复杂的功…...

嵌入式人工智能项目及人工智能应用项目——大合集列表查阅
本文的项目合集列表可能更新不及时(会及时更新),可查阅实时更新的链接如下。 嵌入式人工智能及人工智能应用项目合集实时更新链接如下: 阿齐嵌入式人工智能及人工智能应用项目合集 (kdocs.cn)https://www.kdocs.cn/l/cc97tuieys4…...

心觉:成功学就像一把刀,有什么作用关键在于使用者(一)
Hi,我是心觉,与你一起玩转潜意识、脑波音乐和吸引力法则,轻松掌控自己的人生! 挑战每日一省写作173/1000天 很多人觉得成功学是鸡汤,是没用的,甚至是骗人的 我先保持中立,不知道对不对 我们先…...

GAMES101(10节,几何)
Geometry implicit隐式几何表示: 函数f(x,y,z): 根据函数fn描述几何,遍历所有空间内 的点,如果带入xyz到函数f(x,y,z)结果0那就绘制这个点 如果xyz求值结果>0表示在几何外,0在表面,<0在几何内 构造几何csg(…...
Android 中音频焦点的使用场景及示例
Android 中音频焦点的使用场景及代码示例 一、音频焦点简介 在 Android 系统中,音频焦点(Audio Focus)是一种机制,用于管理多个应用程序同时播放音频时的冲突。当一个应用程序请求音频焦点并获得它时,其他应用程序在…...
2. JDBC驱动是什么?如何在Java项目中配置MySQL的JDBC驱动?
JDBC驱动 是一种软件组件,它使Java应用程序能够与数据库进行交互。JDBC驱动是JDBC API的实现,负责将Java程序中的标准JDBC方法调用转化为数据库特定的操作。每个数据库(如MySQL、PostgreSQL、Oracle等)都有对应的JDBC驱动程序&…...

IDEA运行Tomcat出现乱码问题解决汇总
最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…...
day52 ResNet18 CBAM
在深度学习的旅程中,我们不断探索如何提升模型的性能。今天,我将分享我在 ResNet18 模型中插入 CBAM(Convolutional Block Attention Module)模块,并采用分阶段微调策略的实践过程。通过这个过程,我不仅提升…...

Linux相关概念和易错知识点(42)(TCP的连接管理、可靠性、面临复杂网络的处理)
目录 1.TCP的连接管理机制(1)三次握手①握手过程②对握手过程的理解 (2)四次挥手(3)握手和挥手的触发(4)状态切换①挥手过程中状态的切换②握手过程中状态的切换 2.TCP的可靠性&…...
蓝桥杯 2024 15届国赛 A组 儿童节快乐
P10576 [蓝桥杯 2024 国 A] 儿童节快乐 题目描述 五彩斑斓的气球在蓝天下悠然飘荡,轻快的音乐在耳边持续回荡,小朋友们手牵着手一同畅快欢笑。在这样一片安乐祥和的氛围下,六一来了。 今天是六一儿童节,小蓝老师为了让大家在节…...
条件运算符
C中的三目运算符(也称条件运算符,英文:ternary operator)是一种简洁的条件选择语句,语法如下: 条件表达式 ? 表达式1 : 表达式2• 如果“条件表达式”为true,则整个表达式的结果为“表达式1”…...
使用van-uploader 的UI组件,结合vue2如何实现图片上传组件的封装
以下是基于 vant-ui(适配 Vue2 版本 )实现截图中照片上传预览、删除功能,并封装成可复用组件的完整代码,包含样式和逻辑实现,可直接在 Vue2 项目中使用: 1. 封装的图片上传组件 ImageUploader.vue <te…...
【android bluetooth 框架分析 04】【bt-framework 层详解 1】【BluetoothProperties介绍】
1. BluetoothProperties介绍 libsysprop/srcs/android/sysprop/BluetoothProperties.sysprop BluetoothProperties.sysprop 是 Android AOSP 中的一种 系统属性定义文件(System Property Definition File),用于声明和管理 Bluetooth 模块相…...

企业如何增强终端安全?
在数字化转型加速的今天,企业的业务运行越来越依赖于终端设备。从员工的笔记本电脑、智能手机,到工厂里的物联网设备、智能传感器,这些终端构成了企业与外部世界连接的 “神经末梢”。然而,随着远程办公的常态化和设备接入的爆炸式…...
代码随想录刷题day30
1、零钱兑换II 给你一个整数数组 coins 表示不同面额的硬币,另给一个整数 amount 表示总金额。 请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额,返回 0 。 假设每一种面额的硬币有无限个。 题目数据保证结果符合 32 位带…...

Kafka入门-生产者
生产者 生产者发送流程: 延迟时间为0ms时,也就意味着每当有数据就会直接发送 异步发送API 异步发送和同步发送的不同在于:异步发送不需要等待结果,同步发送必须等待结果才能进行下一步发送。 普通异步发送 首先导入所需的k…...