数据库操作不再困难,MyBatis动态Sql标签解析

系列文章目录

MyBatis缓存原理
Mybatis的CachingExecutor与二级缓存
Mybatis plugin 的使用及原理
MyBatis四大组件Executor、StatementHandler、ParameterHandler、ResultSetHandler 详解
MyBatis+Springboot 启动到SQL执行全流程



在这里插入图片描述

使用MyBatis,或者MyBatis-plus,有一项重要的开发技能就是写动态sql,动态sql能帮我们省略很多复杂逻辑,也能帮我们解决一些格式问题,所以今天我们就来帮大家掌握这个动态sql

📕作者简介:战斧,从事金融IT行业,有着多年一线开发、架构经验;爱好广泛,乐于分享,致力于创作更多高质量内容
📗本文收录于 MyBatis专栏 ,有需要者,可直接订阅专栏实时获取更新
📘高质量专栏 云原生、RabbitMQ、Spring全家桶 等仍在更新,欢迎指导
📙Zookeeper Redis kafka docker netty等诸多框架,以及架构与分布式专题即将上线,敬请期待


一、动态sql是什么?

假设有一个电商网站,需要根据用户提供的不同条件查询商品信息,比如说价格、商品名、类别等等,我们需要写出每个查询信息对应的SQL语句,例如
SELECT * FROM Product WHERE price <= #{price};
SELECT * FROM Product WHERE productname = #{name}
SELECT * FROM Product WHERE type = #{type};

如果这样各种各样的筛选逻辑很多,我们就需要写很多条类似的SQL语句,而且还需要判断入参是否合理,是否为空等等。这样我们的sql就会非常的难以维护。此时,我们就需要利用到MyBatis的动态SQL

MyBatis的动态SQL是指在SQL语句中可以根据不同的条件动态生成不同的SQL语句,以适应不同的需求。动态SQL可以根据条件生成不同的SQL语句,比如WHERE条件、ORDER BY、动态判断等等,从而实现更灵活的SQL编写。动态SQL的主要作用是让SQL语句更加灵活、可重用和易维护,提高应用程序的灵活性和可扩展性

二、动态sql原理

MyBatis动态sql标签的实现原理主要是通过OGNL表达式和Java反射机制来实现的。

首先,MyBatis会将动态sql标签中的OGNL表达式解析成Java代码。解析过程中,MyBatis会根据OGNL表达式的类型来生成相应的Java代码片段。例如,当解析if标签时,如果判断条件中的OGNL表达式为布尔类型,则会生成一个if语句的Java代码片段。

其次,MyBatis会使用Java反射机制获取实体类的属性值,并将属性值传递给动态sql标签中的OGNL表达式进行判断。如果判断条件满足,则会将动态sql标签中的内容添加到SQL语句中;反之,则会忽略动态sql标签中的内容

总体来讲,MyBatis动态sql标签的实现原理就是将OGNL表达式解析成Java代码,并使用Java反射机制获取实体类属性值进行判断,以实现SQL语句的动态生成。myBtais整体运行流程可以看往期的 MyBatis+Springboot 启动到SQL执行全流程

三、动态标签解释

1. if 标签

<if>标签允许你根据条件判断是否包含特定的SQL片段,用于构建灵活的查询语句,该标签的使用方法如下:

  1. 在需要动态生成条件的SQL语句中使用<if>标签。
  2. <if>标签中添加一个test属性,用于指定一个boolean表达式,这个表达式会决定是否将当前条件语句包含在生成的SQL语句中。
  3. <if></if>标签中设置需要动态生成的条件语句。
<select id="getUser" resultMap="userResultMap">SELECT * FROM userwhere 1 = 1<if test="name != null">AND name = #{name}</if><if test="age != null">AND age = #{age}</if>
</select>

在这个示例中,<if>标签用于根据条件动态生成SQL语句中的条件语句。如果传递到这个方法的参数中包含一个非空的name属性,那么就会在SQL语句中添加一个“name = #{name}”的条件语句;如果包含一个非空的age属性,那么就会在SQL语句中添加一个“age = #{age}”的条件语句。

需要注意的是,使用标签时,必须要用“AND”或“OR”将多个<if>标签组合在一起,否则会出现语法错误。此外,还需要注意在使用标签时,SQL语句的正确性和性能的折中。太多的动态条件可能会导致SQL语句生成的复杂度增加,从而影响查询的性能

2 choose、when 和 otherwise 标签

<choose><when><otherwise>标签通常一起使用来实现条件分支查询。这种方式类似于Java中的switch语句或者if-else语句

  • 其中,<choose>标签类似于Java中的switch语句,表示一个选择分支,包含多个<when><otherwise>标签。
  • <when>标签类似于Java中的case语句,用于指定分支条件。当条件满足时,执行<when>标签内的SQL语句。
  • <otherwise>标签类似于Java中的default语句,用于指定默认条件分支。当其他分支条件都不满足时,执行<otherwise>标签内的SQL语句。
<select id="getUsersByAgeAndName" parameterType="map" resultMap="userMap">SELECT * FROM users<where><choose><when test="age != null and name != null">age = #{age} AND name = #{name}</when><when test="age != null">age = #{age}</when><when test="name != null">name = #{name}</when><otherwise>1=1</otherwise></choose></where>
</select>

如上,我们使用<choose><when><otherwise>标签来实现条件分支查询。这个查询可以根据参数中的年龄和名字来查找用户记录,也可以只根据年龄或者名字来查找,或者不加任何条件查询所有的用户记录

3 trim 标签

<trim>标签用于修剪生成的SQL语句,可以用于移除不必要的关键字,特别是在拼接多个条件时非常有用,其主要是用于确定子句中的前缀与后缀。

<trim> 标签具有以下属性:
prefixOverrides:要删除的前缀字符串,可以以“|”分割,添加多个词
suffixOverrides:要删除的后缀字符串,可以以“|”分割,添加多个词
prefix:要添加到 SQL 语句的前缀字符串
suffix:要添加到 SQL 语句的后缀字符串

<select id="findUserByName" parameterType="java.lang.String" resultType="User">SELECT * FROM user<trim prefix="WHERE" prefixOverrides="OR | AND "><if test="name != null and name != ''">AND name = #{name}</if><if test="age != null">AND age = #{age}</if></trim>
</select>

在这个示例中, <trim> 标签用于动态构建 WHERE 子句。如果它会删除前缀中的 OR 或 AND 关键字,并将 WHERE 关键字添加到 SQL 语句中。

当然,你也许会想,如果我把 prefixOverrides 和 prefix 是设置成一个词,那这个词还会不会插入到子句句首呢?答案是会的,因为这里的逻辑是先删除prefixOverrides指定字符串,再添加prefix指定字符串

4 foreach 标签

标签用于循环处理集合类型的参数,生成多次重复的SQL片段,该标签可以用于循环遍历一个集合或数组,并在SQL语句中使用它的值。以下是标签的使用方法:。

遍历集合

<select id="findUserById" parameterType="java.lang.Integer">SELECT * FROM table WHERE id IN<foreach collection="list" item="item" open="(" separator="," close=")">#{item}</foreach>
</select>

其中,collection属性指定要遍历的集合,item属性指定集合中每个元素的别名,open属性指定在开始时添加的字符串,separator属性指定在每个元素之间添加的字符串,close属性指定在结束时添加的字符串。

例如,如果list是一个包含1、2、3的List对象,这个标签会生成以下SQL语句:

SELECT * FROM table WHERE id IN (1,2,3)

遍历数组

<foreach collection="array" item="item" open="(" separator="," close=")">#{item}
</foreach>

这个标签与遍历集合的标签非常相似,只不过把collection属性改为了数组对象

遍历Map

假设我们有这样一个map

Map<String, Object> searchCriteria = new HashMap<>();
searchCriteria.put("username", "john");
searchCriteria.put("age", 25);
searchCriteria.put("city", "New York");

然后我们配上这样的xml

<select id="searchUsers" parameterType="map" resultType="User">SELECT * FROM user<foreach collection="searchCriteria" item="value" index="key" separator="AND" open="WHERE">${key} = #{value}</foreach>
</select>

其中,collection属性指定要遍历的Map集合,index属性指定键的别名,item属性指定值的别名,open属性指定在开始时添加的字符串,separator属性指定在每个元素之间添加的字符串。

最后我们能得到这样的sql

SELECT * FROM user WHERE username = "john" AND age = 25 AND city = "New York"

5 set 标签

标签通常用于动态生成UPDATE语句的SET部分,可自动生成逗号分隔的键值对,<set>标签通常包含多个标签或者标签,用于动态生成要更新的列和值。

  UPDATE user<set><if test="username != null">username = #{username},</if><if test="password != null">password = #{password},</if><if test="email != null">email = #{email},</if></set>WHERE id = #{id}
</update>

在这个示例中,标签根据传入的参数动态生成SET部分。set 元素会动态地在行首插入 SET 关键字,并会删掉额外的逗号

6 bind 标签

<bind>用于将一个表达式绑定到一个变量上。绑定的变量可以在后面的SQL语句中引用。通常,标签被用于简化SQL语句中的重复表达式,提高SQL语句的可读性和可维护性。

<select id="findUsers" parameterType="String" resultMap="userResultMap"><bind name="pattern" value="'%' + name + '%'"/>SELECT * FROM user WHERE name LIKE #{pattern}
</select>

在上面的例子中,标签用于将一个字符串表达式绑定到名为pattern的变量上。变量的值是一个由‘%’,‘name’和‘%’组成的字符串,其中‘name’是一个参数。在后面的SQL语句中,#{pattern}即可引用绑定的变量。

在标签中,name属性指定变量的名称,value属性指定变量的值。可以在value属性中使用表达式,变量名和参数Map中的键。

除了绑定表达式到变量上,标签还可以用于绑定参数值到变量上。例如:

<select id="findUsersByAge" parameterType="int" resultMap="userResultMap"><bind name="minAge" value="age - 5"/><bind name="maxAge" value="age + 5"/>SELECT * FROM user WHERE age BETWEEN #{minAge} AND #{maxAge}
</select>

在上面的例子中,<bind>标签用于将一个整数参数值绑定到名为minAge和maxAge的变量上。变量的值是通过计算参数值得到的。在后面的SQL语句中,#{minAge}和#{maxAge}即可引用绑定的变量。

总之,<bind>标签是一个非常强大的MyBatis功能,可以提高SQL语句的可读性和可维护性。在开发MyBatis应用程序时,应该灵活使用标签,以便更好地管理SQL语句。

7 sql 与 include 标签

<sql>标签用于定义可重用的SQL片段。在一个或多个SQL语句中,可以使用标签来引用这些SQL片段。


<sql id="userColumns">id, username, age
</sql><select id="getAllUsers" resultType="User">SELECT <include refid="userColumns" />FROM user
</select><select id="getUsersByName" parameterType="String" resultType="User">SELECT <include refid="userColumns" />FROM userWHERENAME = #{name}
</select>

上述例子中,<include>标签引用了一个名为"userColumns"的SQL片段。引用的方式是通过refid属性指定片段的ID。所有引用的SQL片段都会被添加到原始SQL语句中。这样,其他sql,如果也是使用这几个字段,则都可以引用

8. where 标签

<where> 标签的作用是用来包裹一个或多个 SQL 条件语句,如果这些条件都满足,则会被加入到 SELECT 或 UPDATE 语句中,否则不会。它的使用方法如下:

1.将多个条件语句用 标签包裹起来。

<select id="selectByCondition" parameterType="java.util.Map" resultMap="userMap">select * from user<where><if test="name != null and name!=''">and name like CONCAT('%',#{name},'%')</if><if test="age != null">and age = #{age}</if></where>
</select>

2.在 SQL 语句中使用 <where> 标签时,会自动移除掉多余的 AND 或 OR,而在没有任何条件语句时,<where> 标签也会自动删除自身,以避免 SQL 语法错误。例如,上述代码在没有任何条件时,生成的 SQL 语句为:

select * from user

而当输入 age=20 时,生成的 SQL 语句为:

select * from userwhere age = 20

可以看出<where> 标签是 MyBatis 中一个非常有用的标签,它可以帮助我们更加方便地动态构建 SQL 语句,提高代码的可读性和可维护性。


总结

MyBatis的动态SQL功能极大地简化了数据库操作,使得针对不同查询需求的SQL语句生成变得灵活而高效。通过深入了解和掌握<if>、<choose>、<when>、<otherwise>、<trim>、<foreach>、<set>、<bind>和<sql>等核心标签,开发人员将大幅度减少因为参数不同,而写不同sql的麻烦,此时的sql会自己按预设逻辑变成不同的样式。因此,需要牢牢掌握。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/69477.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

YOLOv5复现过程出现的问题(关于数据集路径)dataset not found

YOLOv5复现过程出现的问题&#xff08;关于数据集路径&#xff09; 在复现YOLOv5时&#xff0c;按照唐老师的教程&#xff08; https://www.bilibili.com/video/BV11K41167Ar?t122.1&p63&#xff09;下载好了数据集&#xff08;MaskWearing就是检测口罩的一个&#xff0c;…

AT89C51单片机实现单片机串口互动(中断方式,单片机--单片机,应答)

说一下功能&#xff1a;客户机发送0x01到服务机 2服务单片机应答0xf2到客户机 3客户机接收到0xf2,发送信息153432这6个数字到服务机 4client发送完信息后发送0xaa结束通信 5server接收到0xaa后回复0xaa结束通信&#xff0c;从此老死不相往来 看代码&#xff1a; //发送端…

订货系统怎么选?从这四个方面筛选错不了(一)

选择适合的订货系统对企业来说是一个重要且复杂的决策。一个优秀的订货系统可以提高供应链的运作效率、降低成本&#xff0c;并帮助企业更好地管理库存和订单。如果不知道从那几方面做选择&#xff0c;我们可以简单从四个方面进行筛选&#xff0c;这样一般错不了&#xff0c;今…

list

目录 迭代器 介绍 种类 本质 介绍 模拟实现 注意点 代码 迭代器 介绍 在C中&#xff0c;迭代器&#xff08;Iterators&#xff09;是一种用于遍历容器&#xff08;如数组、vector、list等&#xff09;中元素的工具 无论容器的具体实现细节如何,访问容器中的元素的方…

语聚AI如何通过对话方式让AI助手执行应用软件

1 什么是应用助手&#xff1f; 应用助手可以通过对话的方式让AI助手执行应用软件的操作。 例如&#xff0c;您可以调用“Bing搜索引擎搜索”热门信息&#xff0c;然后根据这些信息总结内容。 也可以使用“抖音”应用&#xff0c;搜索热门抖音视频&#xff0c;并根据热门视频生…

Python爱心光波

文章目录 前言Turtle入门简单案例入门函数 爱心光波程序设计程序分析 尾声 前言 七夕要来啦&#xff0c;博主在闲暇之余创作了一个爱心光波&#xff0c;感兴趣的小伙伴们快来看看吧&#xff01; Turtle入门 Turtle 是一个简单而直观的绘图工具&#xff0c;它可以帮助你通过简…

Stable Diffusion WebUI 从零基础到入门

本文主要介绍Stable Diffusion WebUI的实际操作方法&#xff0c;涵盖prompt推导、lora模型、vae模型和controlNet应用等内容&#xff0c;并给出了可操作的文生图、图生图实战示例。适合对Stable Diffusion感兴趣&#xff0c;但又对Stable Diffusion WebUI使用感到困惑的同学&am…

【Apollo】推动创新:探索阿波罗自动驾驶的进步(含安装 Apollo的详细教程)

前言 Apollo (阿波罗)是一个开放的、完整的、安全的平台&#xff0c;将帮助汽车行业及自动驾驶领域的合作伙伴结合车辆和硬件系统&#xff0c;快速搭建一套属于自己的自动驾驶系统。 开放能力、共享资源、加速创新、持续共赢是 Apollo 开放平台的口号。百度把自己所拥有的强大、…

UE4/UE5 照明构建失败 “Lightmass crashed”解决“数组索引越界”

在构建全局光照时,经常会出现“Lightmass crashed”的错误,导致光照构建失败。本文将分析这一问题的原因,并给出解决建议。 UE4 版本4.26 报错如下: <None> === Lightmass crashed: === Assertion failed: (Index >= 0) & (Index < ArrayNum) [File:d:\build…

根据二叉树创建字符串

题目:给你二叉树的根节点 root &#xff0c;请你采用前序遍历的方式&#xff0c;将二叉树转化为一个由括号和整数组成的字符串&#xff0c;返回构造出的字符串。 空节点使用一对空括号对 "()" 表示&#xff0c;转化后需要省略所有不影响字符串与原始二叉树之间的一对…

Linux查看GPU显卡/CPU内存/硬盘信息

显卡信息命令/CPU内存/硬盘 1.显卡2、CPU内存3、硬盘 1.显卡 nvidia-smi nvidia-smi&#xff08;显示一次当前GPU占用情况&#xff09; nvidia-smi -l&#xff08;每秒刷新一次并显示&#xff09; watch -n 5 nvidia-smi &#xff08;其中&#xff0c;5表示每隔6秒刷新一次终端…

达梦数据库(dm8) Centos7 高可用集群

国产数据库-达梦 一、环境详情二、Centos7 参数优化&#xff08;所有节点&#xff09;三、创建用户&#xff08;所有节点&#xff09;四、开始安装&#xff08;所有节点&#xff09;五、服务注册启动 当前安装&#xff1a;在指定版本环境下 测试&#xff0c;仅供参考 官网描述&…