【MyBatis】 MyBatis 动态SQL使用及原理

news2025/1/23 3:52:43

文章目录

  • 前言
  • 1. 动态SQL概述
  • 2. if标签
  • 3. choose、when和otherwise标签
  • 4. trim标签
  • 5. set标签和where标签
  • 6. foreach
  • 7. bind
  • 8. 动态SQL解析原理
  • 总结

前言

MyBatis 是一个优秀的持久层框架,它提供了丰富的 SQL 映射功能,可以让我们通过 XML 或注解方式来定义 SQL 语句。它很大程度上简化了数据库操作,提高了开发效率。动态 SQL 是其中一个非常重要的功能,可以让我们根据不同的条件动态生成 SQL 语句,提高了 SQL 的灵活性和可重用性。本文将详细介绍 MyBatis 的动态 SQL 使用与原理。

1. 动态SQL概述

动态SQL是指根据条件拼接SQL语句的功能,可以在SQL语句中添加或者删除某些条件和语句。在实际开发中,我们经常需要根据不同的条件拼接不同的SQL语句。如果只使用静态SQL,会使得代码冗余度高、可读性差、维护成本高等问题。而使用动态SQL可以很好地解决这些问题。
MyBatis中提供了很多种方式来实现动态SQL,包括if、choose、when、otherwise、trim、where、set等。

2. if标签

if标签是MyBatis中最常用的动态SQL标签之一。它通常用来判断条件是否成立,从而确定是否加入SQL语句中。下面是一段示例代码:

<select id="selectUsers" resultMap="UserResultMap">
    SELECT * FROM Users 
    <where>
        <if test="name != null">
            AND name = #{name}
        </if>
        <if test="age != null">
            AND age = #{age}
        </if>
    </where>
</select>

上述代码中,通过if标签的test属性来判断条件是否成立。只有当"name"和"age"都不为空时,才会将其加入到SQL语句中。这样就可以在不同的情况下生成不同的SQL语句。

3. choose、when和otherwise标签

choose、when和otherwise标签通常一起使用,它类似于Java中的switch语句。下面是一段示例代码:

<select id="selectUsers" resultMap="UserResultMap">
    SELECT * FROM Users 
    <where>
        <choose>
            <when test="name != null">
                AND name = #{name}
            </when>
            <when test="age != null">
                AND age = #{age}
            </when>
            <otherwise>
                AND id > 0
            </otherwise>
        </choose>
    </where>
</select>

choose、when和otherwise标签中,如果test条件成立,就会将当前标签中的SQL语句加入到最终的SQL语句中。只有一个可以成立,多个成立时按顺序第一个生效。

4. trim标签

trim标签通常用来去掉特定字符或者关键字。下面是一段示例代码:

<select id="selectUsers" resultMap="UserResultMap">
    SELECT * FROM Users 
    <where>
        <trim prefix="AND" prefixOverrides="OR">
            <if test="name != null">
                OR name = #{name}
            </if>
            <if test="age != null">
                OR age = #{age}
            </if>
        </trim>
    </where>
</select>

上述代码中,prefix属性表示在标签内部SQL语句前添加的字符;prefixOverrides属性表示从标签内部SQL语句开头去除的字符串。

5. set标签和where标签

set标签通常用来更新参数对象中的非空属性。where标签通常用来拼接SQL语句中的where条件。下面是一段示例代码:

<update id="updateUser" parameterType="User">
    UPDATE Users 
    <set>
        <if test="name != null">
            name = #{name},
        </if>
        <if test="age != null">
            age = #{age},
        </if>
    </set>
    <where>
        id = #{id}
    </where>
</update>

上述代码中,set标签用来设置要更新的字段,通过if标签判断哪些字段需要更新。where标签用来拼接SQL语句中的where条件,具体的条件可以根据实际情况进行调整。

6. foreach

foreach 标签用于处理集合类型的参数,比如 List、Array 等,可以遍历集合中的元素,将每个元素都转化为 SQL 语句的一部分,用于生成动态 SQL 语句。下面是一个示例:

<select id="getUserByIdList" resultType="User">
    SELECT * FROM user
    WHERE id IN
    <foreach collection="idList" item="id" open="(" separator="," close=")">
        #{id}
    </foreach>
</select>

在上述 SQL 语句中,我们通过 foreach 标签遍历传入的参数 idList,将其中的每个元素转化为一个 id,然后根据这些 id 拼接成一个 IN 子句。

7. bind

bind 标签用于定义一个变量,该变量可以被后续的 SQL 片段引用,方便了 SQL 的编写。下面是一个示例:

<select id="getUserByName" resultType="User">
    <bind name="queryName" value="'%' + name + '%'"/>
    SELECT * FROM user WHERE name like #{queryName}
</select>

在上述 SQL 语句中,我们使用 bind 标签定义了一个变量 queryName,它的值为 name 模糊查询的条件。然后使用该变量来拼接 SQL 语句,使得 SQL 语句更加简洁。

8. 动态SQL解析原理

MyBatis的动态SQL是通过OGNL表达式来实现的。OGNL(Object-Graph Navigation Language)是一种基于Java对象图遍历的表达式语言,它可以方便地访问Java对象的属性和方法。

在MyBatis中,通过OGNL表达式可以动态地计算条件是否成立,从而确定是否将SQL片段添加到最终的SQL语句中。OGNL表达式通常嵌入在MyBatis中的动态SQL标签中,例如if、choose、when、otherwise等。

MyBatis使用了两个重要的类来实现OGNL表达式的解析和计算:OgnlExpressionEvaluator和OgnlCache。OgnlExpressionEvaluator类负责将MyBatis传入的参数对象转换为OGNL表达式需要的上下文对象,然后将OGNL表达式计算结果返回;OgnlCache类负责缓存已经解析好的OGNL表达式,避免重复解析和计算。

具体的解析过程如下:

  1. 根据MyBatis的配置将Mapper.xml文件中的SQL语句解析为一个MappedStatement对象,并将其中的OGNL表达式解析成一个一个可执行的SQL片段。

  2. 对于每一个OGNL表达式,MyBatis使用${}来表示一个简单的OGNL表达式,使用#{}来表示一个OGNL表达式中包含复杂逻辑的情况。在解析过程中,MyBatis会将OGNL表达式中的参数进行解析和预处理,然后使用OgnlCache类将其缓存起来。

  3. 当Mapper接口方法被调用时,MyBatis会将方法中传入的参数对象转换为一个BoundSql对象,并将该BoundSql对象与MappedStatement对象一起传递给OgnlExpressionEvaluator类。

  4. OgnlExpressionEvaluator类中再次解析OGNL表达式,并将BoundSql对象作为上下文传入OGNL表达式中执行。OGNL表达式执行的结果将被转化为String类型,并返回给BoundSql对象。

  5. 最后,MyBatis将所有BoundSql对象中的SQL片段拼接成最终的SQL语句并执行。

MyBatis的动态SQL解析原理是将OGNL表达式解析为可执行的SQL片段,然后根据条件判断是否将该SQL片段加入到最终的SQL语句中。MyBatis使用OgnlExpressionEvaluator和OgnlCache类来实现OGNL表达式的解析和计算,从而实现动态SQL的功能。

在 MyBatis 的源码中,动态 SQL 还涉及到以下接口和类来实现:

  1. SqlNode 接口:表示一个 SQL 节点,也就是一个 SQL 片段。它包含一个 apply 方法,在执行 SQL 语句时会将 SQL 片段应用到相应的位置。

  2. MixedSqlNode 类:实现了 SqlNode 接口,可以包含多个子节点。该类的 apply 方法会依次遍历所有子节点,并将每个节点应用到 SQL 语句中。

  3. TextSqlNode 类:表示一个纯文本节点。该类包含一个文本字符串,可以将其直接应用到 SQL 语句中。

  4. IfSqlNode 类:表示一个条件节点。可以根据指定的条件判断是否需要应用该节点内部的 SQL 片段。如果条件成立,则会将 SQL 片段应用到 SQL 语句中。

  5. TrimSqlNode 类:表示一个修剪节点,可以根据配置对 SQL 片段进行修剪操作。常用于处理 UPDATE 和 INSERT 语句中 SET 子句的逗号问题。

  6. WhereSqlNode 类:表示一个 WHERE 条件节点。可以将 WHERE 子句的参数拼接到 SQL 语句中。

在这里插入图片描述

以上是 MyBatis 中实现动态 SQL 的核心接口和类。MyBatis 内部通过组合这些接口和类来构建复杂的 SQL 语句。通过定义这些接口和类,可以让开发者更加方便地书写动态 SQL 语句,并且遵循了设计模式中的单一职责原则。

还有一些Builder 接口及其实现类的作用都是用于构造 SQL 语句。下面简单介绍一下一些常用的 Builder 类型:

  1. BaseBuilder 接口:所有 Builder 的基础接口,定义了一些共同的方法,例如获取 Configuration 对象、创建 ParameterMapping 对象等。

  2. XMLMapperBuilder 类:从 XML 文件中解析出各种 SQL 节点,然后通过其他 Builder 对象将其转换成 SQL 语句。

  3. MapperBuilderAssistant 类:辅助 XMLMapperBuilder 类创建各种类型的 SQL 节点,例如创建 <select>、<update>、<insert> 等标签节点。

  4. SqlSourceBuilder 类:根据 XML 中的 SQL 片段创建 SqlSource 对象,SqlSource 对象中包含了解析后的 SQL 语句和参数信息。

  5. DynamicSqlSource 类:用于处理动态 SQL,也就是包含各种条件判断和循环语句的 SQL 片段。它是 SqlSource 接口的一种实现。

  6. StaticSqlSource 类:用于处理静态 SQL,即不包含任何条件语句和循环语句的 SQL 片段。它同样是 SqlSource 接口的一种实现。

  7. SqlSessionFactoryBuilder 类:用于创建 SqlSessionFactory 对象,它会将所有的 Builder 对象组合在一起,完成 SQL 语句的解析和构造。

在这里插入图片描述

通过上述不同类型的 Builder 对象,我们可以将 XML 中的 SQL 片段转换成 Java 对象,并且根据各种条件生成相应的 SQL 语句。这个过程中涉及到的类和方法非常多,需要我们深入地了解 MyBatis 的内部实现才能灵活运用。

总结

本文通过介绍MyBatis动态SQL的基本概念和常用标签(if、choose、when、otherwise、trim、where、set、foreach),希望读者能够更加深入地了解MyBatis的使用和原理。在实际开发过程中,要根据具体场景和需求选择合适的动态SQL标签,从而实现灵活拼接SQL语句的功能,提高开发效率。
在 MyBatis 中,动态 SQL 主要包括以下几种类型:

  1. <if> 标签:表示一个条件语句,可以根据条件判断是否包含相应的 SQL 片段。
  2. <where> 标签:表示一个 WHERE 条件语句,可以根据配置自动添加 WHERE 关键字。
  3. <choose> 标签:表示一个选择语句,可以根据多个条件选择符合条件的 SQL 片段。
  4. <foreach> 标签:表示一个循环语句,在循环中动态生成 SQL 语句。
  5. <set> 标签:表示一个 SET 子句,可以根据指定的属性值动态生成 SET 语句。

以上标签都属于动态 SQL,在解析时需要通过特殊的方式进行处理。下面以<if> 标签为例介绍解析原理:

  1. XMLScriptBuilder 类会根据标签类型创建相应的 SQL 节点,例如 <if> 标签对应的节点是 IfSqlNode 对象。
  2. XMLScriptBuilder 类会递归解析节点内部的子节点,并将其组合成一个 SQL 片段。
  3. 当解析到 IfSqlNode 节点时,XMLScriptBuilder 类会获取标签中的 test 属性,并根据该属性值创建一个 OgnlExpression 对象(OGNL 表达式对象),用于判断条件是否满足。
  4. 如果条件满足,则将子节点生成的 SQL 片段添加到当前 SQL 上下文中;否则忽略该节点。
  5. 最终生成的 SQL 语句就是将所有满足条件的 SQL 片段组合起来得到的。

以上就是 MyBatis 实现动态 SQL 解析的大体流程。通过 XMLScriptBuilder 类的递归解析,可以将各种类型的动态 SQL 节点转换成 SqlNode 接口的实现,然后通过 MixedSqlNode 类将它们组合成一个完整的 SQL 片段。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

@SpringBootApplication深入剖析

如下图 SpringBootApplication是springboot项目启动类的注解&#xff0c;也是程序的入口,本文就是具体解析一下这个注解到底做了什么 一.SpringBootApplication的构成 1.这个注解是一个组合注解&#xff0c;他是有三个注解合成的&#xff0c;对应图中的123步&#xff0c;而这三…

新的勒索软件加密自身以逃避防病毒

一种名为 Cactus 的新型勒索软件一直在利用 VPN 设备中的漏洞对“大型商业实体”的网络进行初始访问。 Cactus 勒索软件行动至少从 3 月开始就一直活跃&#xff0c;并正在寻求受害者的大笔支出。 虽然新的威胁参与者采用了勒索软件攻击中常见的策略——文件加密和数据窃取——…

大数据Doris(十七):Random Distribution和复合分区使用场景

文章目录 Random Distribution和复合分区使用场景 一、Random Distribution 二、复合分区使用场景 Random Distribution和复合分区使用场景 一、Random Distribution 如果 OLAP 表没有更新类型的字段&#xff0c;将表的数据分桶模式设置为 RANDOM&#xff0c;则可以避免严…

架构设计之需求分析

大家好&#xff0c;我是易安。 设计架构的第一步是需求分析。那么&#xff0c;为什么要做需求分析&#xff1f;如何做好需求分析&#xff1f;今天我们一起聊一聊需求分析这件事儿 为什么要做需求分析 为何要做需求分析&#xff1f; 首先&#xff0c;当然是因为我们做软件本身就…

迭代器失效问题,以及解决方法。

迭代器的主要作用就是让算法能够不用关心底层数据结构&#xff0c;其底层实际就是一个指针&#xff0c;或者是对指针进行了封装&#xff0c;比如&#xff1a;vector的迭代器就是原生态指针T* 。因此迭代器失效&#xff0c;实际就是迭代器底层对应指针所指向的空间被销毁了&…

【小沐学Python】Python实现Web服务器(Flask+Vue+node.js,web单页增删改查)

文章目录 1、简介1.1 flask1.2 vue 2、开发2.1 新建flask项目2.2 安装flask库2.3 新建flask的主脚本2.4 新建Vue项目2.5 安装vue项目依赖项2.6 新增组件Ping.vue2.7 Ping.vue增加HTTP请求2.8 美化vue前端页面2.9 新增组件Books.vue2.10 flask增加路由Books2.11 Books.vue增加HT…

什么是ChatGPT?怎么用?

最近全网爆火的黑科技&#xff0c;叫做chatGPT。ChatGPT声称&#xff0c;它的AI对话模型能在大范围、细粒度问题上给出普遍准确的答案。简单地说&#xff0c;AI对话模型可以达到基本不犯错误的水平了。那么到底这个ChatGPT是什么&#xff1f;怎么用&#xff1f;本篇文章就来带大…

算法修炼之练气篇——练气二层

博主&#xff1a;命运之光 专栏&#xff1a;算法修炼之练气篇 题目 1084: 用筛法求之N内的素数 题目描述 用筛法求之N内的素数。 输入格式 N 输出格式 0&#xff5e;N的素数 样例输入 100 样例输出 2 3 5 7 11 13 17 19 23 29 31 37 41 43 47 53 59 61 67 71 73 79 …

学系统集成项目管理工程师(中项)系列21a_整体管理(上)

1. 含义 1.1. 包括为识别、定义、组合、统一和协调各项目管理过程组的各种过程和活动而开展的工作&#xff0c;是项目管理中一项综合性和全局性的管理工作 2. 项目经理是整合者 2.1. 【21上选33】 2.1.1. 【19上选37】 2.1.2. 【22上选33】 2.2. 通过与项目干系人主动、全…

shell脚本(磁盘空间、服务状态)

1、判断当前磁盘剩余空间是否有20G&#xff0c;如果小于20G&#xff0c;则将报警邮件发送给管理员&#xff0c;每天检查一次磁盘剩余空间。 第一步&#xff1a;创建脚本名为shell1.sh如下&#xff1a; vim shell1.sh 第二步&#xff1a;做计划在shell1文件中&#xff0c;命令…

Kyligence Zen 简直就是一站式指标平台的天花板

一、Kyligence Zen是什么&#xff1f; 1、Kyligence Zen是做啥的&#xff1f; Kyligence Zen是一款指标分析和管理的工具&#xff0c;是基于 Kyligence 核心 OLAP 能力打造&#xff0c;Kyligence Zen 提供集业务模型、指标管理、指标加工、数据服务于一体的一站式服务&#x…

孙溟㠭20余载春秋,4000多方印章,这双质朴的手有多么倔强的生命力

作品的背后往往折射出艺术家人生的广度和厚度。 先锋篆刻、书画艺术家孙溟㠭&#xff0c; 上世纪90年代开始接触篆刻&#xff0c; 至今&#xff0c;20载有余&#xff0c;积累了4000多方篆刻作品。 在他创作纪念吴品超院士的作品《药生尘》时&#xff0c; 我们拍到了艺术家…

高级Web题库

高级Web题库 For ZPT 声明 一切开发旨在学习&#xff0c;请勿用于非法用途 by rick rick 关注 永雏塔菲喵 永雏塔菲喵 选择题 第1题 知识点&#xff1a;CSS 题目&#xff1a;设置text-decoration属性的删除线的值为&#xff08; &#xff09;。 选项&#xff1a; A underlin…

固定翼无人机培训第二周总结——多轴和起降

博主学的III类固定翼垂直起降无人机&#xff0c;起降采用多旋翼&#xff08;下图中红框就是旋翼&#xff09;&#xff0c;巡航采用固定翼。 理论大部分也是多旋翼&#xff0c;多轴旋翼无人机是指三个旋翼轴及以上的特殊直升机&#xff0c;多旋翼无人机靠旋翼速度和方向来控制无…

代码随想录算法训练营第三十八天 | 动态规划基础流程

动态规划理论基础 代码随想录 (programmercarl.com) 如果某一问题有很多重叠子问题&#xff0c;使用动态规划是最有效的。 所以动态规划中每一个状态一定是由上一个状态推导出来的&#xff0c;这一点就区分于贪心&#xff0c;贪心没有状态推导&#xff0c;而是从局部直接选最…

Java结合POI框架实现Excel导入

Java结合POI框架实现Excel导入 一、流程概念二、conroller中的方法三、导入成功 一、流程概念 我们需要把excel通过上传得方式导入数据库&#xff0c;需要以下几个步骤 将excel上传到服务器指定文件夹内并重命名&#xff08;upload&#xff09;获取到文件公共路径和别名路径将…

InsCode再进步,AI 辅助编程帮你打开思路

文章目录 一、前言二、使用 AI 辅助完成代码1. 基于模板创建项目2. 使用 AI 辅助开拓思路3. 使用 AI 辅助生成代码4. 使用 AI 辅助优化代码 三、InsCode AI Chat 的使用建议四、总结 一、前言 你好&#xff0c;我是小雨青年&#xff0c;一名独立开发的程序员。 在之前的文章中…

Ubuntu22.04安装PyTorch1.13.0 GPU版本

目录 一、电脑相关信息 1. 电脑显卡环境&#xff1a; 二、安装Pytorch1.13.0/cu117&#xff08;GPU版本&#xff09; 1. 准备&#xff1a;新建虚拟环境 2. 用conda在线安装pytorch1.13.0/cu117&#xff08;pytorch1.13.0 torchvision0.14.0 pytorch-cuda11.7&#xff09;…

博客管理系统前端分析

目录结构博客列表页&#xff1a;所有页面共同的样式代码&#xff1a;博客详情页博客登录页博客编辑页 目录结构 博客列表页&#xff1a; 页面效果&#xff1a; 代码&#xff1a; <!-- 博客列表页 --> <!DOCTYPE html> <html lang"en"> <head…

计算机视觉的深度学习 Lecture4:Optimization 笔记 EECS 498.008

数值计算梯度 问题是慢&#xff0c;每个都要注意做步长&#xff0c;求除法。 应该用求导方法解决。 SGD通过每次抽取一部分&#xff08;mini-batch&#xff09;来计算梯度&#xff0c;而不是遍历整个数据集来求梯度&#xff0c;大大增大了求梯度速度&#xff0c;并且性能不…