您当前的位置:首页 > 电脑百科 > 程序开发 > 框架

简单学习一下 MyBatis 动态SQL使用及原理

时间:2023-05-12 15:37:42  来源:稀土掘金  作者:互联网高级架构师


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

一、动态SQL概述

动态SQL是指根据条件拼接SQL语句的功能,可以在SQL语句中添加或者删除某些条件和语句。在实际开发中,我们经常需要根据不同的条件拼接不同的SQL语句。如果只使用静态SQL,会使得代码冗余度高、可读性差、维护成本高等问题。而使用动态SQL可以很好地解决这些问题。

MyBatis中提供了很多种方式来实现动态SQL,包括if、choose、when、otherwise、trim、where、set等。

二、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语句。

三、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语句中。只有一个可以成立,多个成立时按顺序第一个生效。

四、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语句开头去除的字符串。

五、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条件,具体的条件可以根据实际情况进行调整。

六、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 子句。

七、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 语句更加简洁。

八、动态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 片段。

作者:Cosolar
链接:
https://juejin.cn/post/7231921877466677285
来源:稀土掘金



Tags:MyBatis   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,不构成投资建议。投资者据此操作,风险自担。如有任何标注错误或版权侵犯请与我们联系,我们将及时更正、删除。
▌相关推荐
Mybatis参数映射搞不明白?来试试这个工具吧!
之前在《使用技巧-Mybatis参数映射》《使用技巧-Mybatis参数映射(2)》提到了Mybatis的一些参数映射技巧,但是平时使用的时候有些小伙伴可能不知道自己写的#{}表达式能不能获取...【详细内容】
2024-02-28  Search: MyBatis  点击:(33)  评论:(0)  加入收藏
如何在Spring项目中配置MP(MyBatis-Plus)集成?
在Spring项目中集成MP,需要进行以下配置:1. 引入依赖:在项目的pom.xml文件中添加MP相关依赖,例如:```xml<dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plu...【详细内容】
2024-01-09  Search: MyBatis  点击:(87)  评论:(0)  加入收藏
Mybatis占位符#和$的区别?源码解读
本文针对笔者日常开发中对 Mybatis 占位符 #{} 和 ${} 使用时机结合源码,思考总结而来 &bull; Mybatis 版本 3.5.11 &bull; Spring boot 版本 3.0.2 &bull; mybatis-spring...【详细内容】
2023-10-27  Search: MyBatis  点击:(399)  评论:(0)  加入收藏
看完这篇文章,你也可以手写MyBatis部分源码(JDBC)
一、持久化机制持久化(persistence): 把数据保存到可调电式存储设备中以供之后使用。大多数情况下,特别是企业级应用,数据持久化意味着将内存中的数据保存到硬盘上加以”固化...【详细内容】
2023-10-09  Search: MyBatis  点击:(332)  评论:(0)  加入收藏
Mybatis-Flex初体验
本篇文章内容主要包括: MyBatis-Flex 介绍MyBatis-Flex 是一个优雅的 MyBatis 增强框架,它非常轻量、同时拥有极高的性能与灵活性。我们可以轻松的使用 Mybaits-Flex 链接任何...【详细内容】
2023-09-24  Search: MyBatis  点击:(196)  评论:(0)  加入收藏
MyBatis简单易用的背后隐藏的挑战
MyBatis,作为一款备受欢迎的持久层框架,它的简单易用以及灵活的配置吸引了无数的开发者。然而,随着项目的不断发展,规模的逐渐扩大,MyBatis的一些挑战也开始逐渐浮出水面。首先,由...【详细内容】
2023-09-15  Search: MyBatis  点击:(242)  评论:(0)  加入收藏
MyBatis缓存机制
MyBatis 的缓存机制属于本地缓存,适用于单机系统,它的作用是减少数据库的查询次数,提高系统性能。MyBaits 中包含两级本地缓存: 一级缓存:SqlSession 级别的,是 MyBatis 自带的缓...【详细内容】
2023-09-12  Search: MyBatis  点击:(229)  评论:(0)  加入收藏
对比 MyBatis 和 MyBatis-Plus 批量插入、批量更新的性能和区别
1 环境准备1.1 搭建 MyBatis-Plus 环境 创建 maven springboot 工程 导入依赖:web 启动器、jdbc、、java 连接 mysql、Lombok、druid 连接池启动器、mybatis-plus 启动器 编...【详细内容】
2023-09-08  Search: MyBatis  点击:(191)  评论:(0)  加入收藏
Spring Data JPA 和 MyBatis 谁更强?
我无法明确的告诉你JPA和MyBatis在国内哪个会更流行,我本人更喜欢JPA,但是我本人日常开发用MyBatis多。但是我的回答绝对不是在划水,而是我多年来自己的一点小小的思考。MyBati...【详细内容】
2023-08-22  Search: MyBatis  点击:(335)  评论:(0)  加入收藏
Mybatis-Plus可能会导致数据库死锁
一、场景还原1.版本信息MySQL版本:5.6.36-82.1-logMybatis-Plus的starter版本:3.3.2存储引擎:InnoDB2.死锁现象A同学在生产环境使用了Mybatis-Plus提供的com.baomidou.mybatisp...【详细内容】
2023-08-14  Search: MyBatis  点击:(171)  评论:(0)  加入收藏
▌简易百科推荐
Web Components实践:如何搭建一个框架无关的AI组件库
一、让人又爱又恨的Web ComponentsWeb Components是一种用于构建可重用的Web元素的技术。它允许开发者创建自定义的HTML元素,这些元素可以在不同的Web应用程序中重复使用,并且...【详细内容】
2024-04-03  京东云开发者    Tags:Web Components   点击:(8)  评论:(0)  加入收藏
Kubernetes 集群 CPU 使用率只有 13% :这下大家该知道如何省钱了
作者 | THE STACK译者 | 刘雅梦策划 | Tina根据 CAST AI 对 4000 个 Kubernetes 集群的分析,Kubernetes 集群通常只使用 13% 的 CPU 和平均 20% 的内存,这表明存在严重的过度...【详细内容】
2024-03-08  InfoQ    Tags:Kubernetes   点击:(12)  评论:(0)  加入收藏
Spring Security:保障应用安全的利器
SpringSecurity作为一个功能强大的安全框架,为Java应用程序提供了全面的安全保障,包括认证、授权、防护和集成等方面。本文将介绍SpringSecurity在这些方面的特性和优势,以及它...【详细内容】
2024-02-27  风舞凋零叶    Tags:Spring Security   点击:(54)  评论:(0)  加入收藏
五大跨平台桌面应用开发框架:Electron、Tauri、Flutter等
一、什么是跨平台桌面应用开发框架跨平台桌面应用开发框架是一种工具或框架,它允许开发者使用一种统一的代码库或语言来创建能够在多个操作系统上运行的桌面应用程序。传统上...【详细内容】
2024-02-26  贝格前端工场    Tags:框架   点击:(47)  评论:(0)  加入收藏
Spring Security权限控制框架使用指南
在常用的后台管理系统中,通常都会有访问权限控制的需求,用于限制不同人员对于接口的访问能力,如果用户不具备指定的权限,则不能访问某些接口。本文将用 waynboot-mall 项目举例...【详细内容】
2024-02-19  程序员wayn  微信公众号  Tags:Spring   点击:(39)  评论:(0)  加入收藏
开发者的Kubernetes懒人指南
你可以将本文作为开发者快速了解 Kubernetes 的指南。从基础知识到更高级的主题,如 Helm Chart,以及所有这些如何影响你作为开发者。译自Kubernetes for Lazy Developers。作...【详细内容】
2024-02-01  云云众生s  微信公众号  Tags:Kubernetes   点击:(50)  评论:(0)  加入收藏
链世界:一种简单而有效的人类行为Agent模型强化学习框架
强化学习是一种机器学习的方法,它通过让智能体(Agent)与环境交互,从而学习如何选择最优的行动来最大化累积的奖励。强化学习在许多领域都有广泛的应用,例如游戏、机器人、自动驾...【详细内容】
2024-01-30  大噬元兽  微信公众号  Tags:框架   点击:(68)  评论:(0)  加入收藏
Spring实现Kafka重试Topic,真的太香了
概述Kafka的强大功能之一是每个分区都有一个Consumer的偏移值。该偏移值是消费者将读取的下一条消息的值。可以自动或手动增加该值。如果我们由于错误而无法处理消息并想重...【详细内容】
2024-01-26  HELLO程序员  微信公众号  Tags:Spring   点击:(86)  评论:(0)  加入收藏
SpringBoot如何实现缓存预热?
缓存预热是指在 Spring Boot 项目启动时,预先将数据加载到缓存系统(如 Redis)中的一种机制。那么问题来了,在 Spring Boot 项目启动之后,在什么时候?在哪里可以将数据加载到缓存系...【详细内容】
2024-01-19   Java中文社群  微信公众号  Tags:SpringBoot   点击:(86)  评论:(0)  加入收藏
花 15 分钟把 Express.js 搞明白,全栈没有那么难
Express 是老牌的 Node.js 框架,以简单和轻量著称,几行代码就可以启动一个 HTTP 服务器。市面上主流的 Node.js 框架,如 Egg.js、Nest.js 等都与 Express 息息相关。Express 框...【详细内容】
2024-01-16  程序员成功  微信公众号  Tags:Express.js   点击:(88)  评论:(0)  加入收藏
站内最新
站内热门
站内头条