动态 SQL 是 MyBatis 的强大特性之一。如果你使用过 JDBC 或其它类似的框架,你应该能理解根据不同条件拼接 SQL 语句有多痛苦,例如拼接时要确保不能忘记添加必要的空格,还要注意去掉列表最后一个列名的逗号。利用动态 SQL,可以彻底摆脱这种痛苦。

如果你之前用过 JSTL 或任何基于类 XML 语言的文本处理器,你对动态 SQL 元素可能会感觉似曾相识。在 MyBatis 之前的版本中,需要花时间了解大量的元素。借助功能强大的基于 OGNL 的表达式,MyBatis 3 替换了之前的大部分元素,大大精简了元素种类,现在要学习的元素种类比原来的一半还要少。

  • if
  • choose (when, otherwise)
  • trim (where, set)
  • foreach

理解:在sql层面执行逻辑代码

  1. if

    根据不同的条件(包括无条件)进行查询,大大降低了代码的冗余

        <select id="queryByIf" parameterType="map" resultType="userBean">
    SELECT * FROM user WHERE 1=1
    <if test="UId != null">
    AND UId = #{UId}
    </if>
    <if test="UName != null">
    AND UName = #{UName}
    </if>
    </select>

    注:(1).这里resultType写的是集合中的元素类型,并不是集合本身

    (2).test中的条件可以通过and连接

  2. chose,when,otherwise

    我们不想使用所有的条件,而只是想从多个条件中选择一个使用。

        <select id="queryByIf" parameterType="map" resultType="userBean">
    SELECT * FROM user WHERE
    <choose>
    <when test="UId != null">
    UId = #{UId}
    </when>
    <when test="UName != null">
    UName = #{UName}
    </when>
    <otherwise>
    1=1
    </otherwise>
    </choose>
    </select>

    注:这个语句类似swith,choose->switch,when->case,otherwise->default

  3. trim、where、set

    where 元素只会在子元素返回任何内容的情况下才插入 “WHERE” 子句。而且,若子句的开头为 “AND” 或 “OR”,where 元素也会将它们去除。

    set 元素会动态地在行首插入 SET 关键字,并会删掉额外的逗号(这些逗号是在使用条件语句给列赋值时引入的)。

    trim可以完成对两者的实现与优化

    where使用:

        <select id="queryByIf" parameterType="map" resultType="userBean">
    SELECT * FROM user
    <where>
    <if test="UName != null">
    AND UState = #{UState}
    </if>
    <choose>
    <when test="UId != null">
    AND UId = #{UId}
    </when>
    <otherwise>
    AND 1=1
    </otherwise>
    </choose>
    </where>
    </select>

    注:这里的where写了之后,原句中的where就不用写了,在代码运行过程,where标签能够自动补上where,并且去除不需要的AND(不会补上)

    set使用:

        <update id="updateById" parameterType="map">
    UPDATE user
    <set>
    <if test="UState != null">UState = #{UState},</if>
    <choose>
    <when test="UName != null">
    UName = #{UName},
    </when>
    <when test="USet != null">
    USet = #{USet},
    </when>
    </choose>
    </set>
    WHERE UId = #{UId}
    </update>

    注:这里的set写了之后,原句中的set也会自动补全,并且去除不需要的','(不会补上)

    trim的定义格式:

    实现where

    <trim prefix="WHERE" prefixOverrides="AND |OR ">
    ...
    </trim>

    实现set

    <trim prefix="SET" suffixOverrides=",">
    ...
    </trim>
  4. foreach使用

    动态 SQL 的另一个常见使用场景是对集合进行遍历(尤其是在构建 IN 条件语句的时候)

        <select id="queryListByForeach" parameterType="map" resultType="userBean">
    SELECT * FROM user
    <where>
    <foreach collection="USets" item="item" open="(" separator="OR" close=")">
    USet = #{item}
    </foreach>
    </where>
    </select>

    说明:主要解释一下foreach中的几个参数,这里collection是map中的一个集合,Uset为集合中的一个参数(名字随意),open定义了迭代开始所放置的字符串,separator定义的是迭代时的分隔符,close定义了迭代结束放置的字符串

  5. 补充:sql片段

    类似于前端组件的概念

    通过<sql id="">定义组件,通过<include refid="">引用组件

    注意:

    (1). 最好基于单表来定义SQL片段

    (2). 不要存在where标签

Mybaties——动态sql的更多相关文章

  1. 小D课堂-SpringBoot 2.x微信支付在线教育网站项目实战_3-4.动态Sql语句Mybaties SqlProvider

    笔记 4.动态Sql语句Mybaties SqlProvider     简介:讲解什么是动态sql,及使用 1.             @UpdateProvider(type=VideoSqlP ...

  2. 值得注意的ibatis动态sql语法格式

    一.Ibatis常用动态sql语法,简单粗暴用一例子 <select id="iBatisSelectList" parameterClass="java.util ...

  3. Mysql - 游标/动态sql/事务

    游标这个在我目前的项目里面用的还不多, 但是其功能还是很强大的. 动态sql以前都没用过, 是跟着富士康(不是张全蛋的富土康哦)过来的同事学的. 还是挺好用的. 我的数据库方面, 跟他学了不少. 在此 ...

  4. MyBatis4:动态SQL

    什么是动态SQL MyBatis的一个强大特性之一通常是它的动态SQL能力.如果你有使用JDBC或其他相似框架的经验,你就明白条件串联SQL字符串在一起是多么地痛苦,确保不能忘了空格或者在列表的最后的 ...

  5. 分享公司DAO层动态SQL的一些封装

    主题 公司在DAO层使用的框架是Spring Data JPA,这个框架很好用,基本不需要自己写SQL或者HQL就能完成大部分事情,但是偶尔有一些复杂的查询还是需要自己手写原生的Native SQL或 ...

  6. MySQL存储过程动态SQL语句的生成

    用Mysql存储过程来完成动态SQL语句,使用存储过程有很好的执行效率: 现在有要求如下:根据输入的年份.国家.节假日类型查询一个节假日,我们可以使用一般的SQL语句嵌入到Java代码中,但是执行效率 ...

  7. 【Java EE 学习 79 下】【动态SQL】【mybatis和spring的整合】

    一.动态SQL 什么是动态SQL,就是在不同的条件下,sql语句不相同的意思,曾经在“酒店会员管理系统”中写过大量的多条件查询,那是在SSH的环境中,所以只能在代码中进行判断,以下是其中一个多条件查询 ...

  8. 自定义函数执行动态sql语句

    --函数中不能调用动态SQL,使用用存储过程吧.如果还要对函数做其他操作,换成存储过程不方便,可以考虑把其他操作一起封装在存储过程里面.如:   create proc [dbo].[FUN_YSCL ...

  9. mybatis入门基础(五)----动态SQL

    一:动态SQL 1.1.定义 mybatis核心对sql语句进行灵活操作,通过表达式进行判断,对sql进行灵活拼接.组装. 1.2.案例需求 用户信息综合查询列表这个statement的定义使用动态s ...

随机推荐

  1. Python语言编程基础

    Python 技能目标 理解编程基本原理和思想 掌握python语言的基础语法 能够使用python进行基本的开发工作 熟练使用项目开发IDE:eclipse,PyDev 能够使用Python开发简单 ...

  2. 北京太速-611号-基于VU9P的5Gsps高速ADDA收发PCIe卡

    1    板卡概述 基于XCVU9P的5Gsps AD DA收发PCIe板卡.该板卡要求符合PCIe 3.0标准,包含一片XCVU9P-2FLGA2014I.2组64-bit/8GB DDR4.2路高 ...

  3. 【论文考古】知识蒸馏 Distilling the Knowledge in a Neural Network

    论文内容 G. Hinton, O. Vinyals, and J. Dean, "Distilling the Knowledge in a Neural Network." 2 ...

  4. Redis——入门学习笔记

    Redis学习 说到前面:这篇笔记只是我作为一个Redis新手,从0到认知的一个过程.后续会持续深入学习. 学习初衷和计划 学习Redis,因为这是热门技术,必须掌握的技术,别人都会我不会.就这一点就 ...

  5. Spring 类名后缀理解

    Aware 理解 实现Spring的Aware接口. 定义为感知.意识,核心意义在于通过Aware可以把spring底层组件注入到自定义的bean中. 对于bean与容器的关系,bean不应该知道自身 ...

  6. Ansible部署K8s集群

    目录 检查网络:k8s-check.yaml 连接配置:k8s-conn-cfg.yaml 配置k8s集群dns解析: k8s-hosts-cfg.yaml 配置yum源:k8s-yum-cfg.ya ...

  7. ssh远程端口转发&&windows系统提权之信息收集&&网安工具分享(部分)

    一.ssh远程端口转发 背景:当我们在渗透过程中,获取到内网的一台仅有内网IP的服务器后,我们可以通过ssh隧道,将内网某个主机的端口进行远程转发 1.网络拓扑图 假设获取的服务器为web服务器,we ...

  8. flask 中使用蓝图将路由分开写在不同文件

    flask 若想将不同的路由写在不同的文件中(如将 user 对象的相关接口写在一个文件中,将 customer 对象的相关接口写在另一个文件中),可以使用蓝图来实现. 有关蓝图的定义:A Bluep ...

  9. [题解]Mail.Ru Cup 2018 Round 1 - A. Elevator or Stairs?

    [题目] A. Elevator or Stairs? [描述] Masha要从第x层楼去第y层楼找Egor,可以选择爬楼梯或者坐直升电梯.已知爬楼梯每层需要时间t1:坐直升电梯每层需要时间t2,直升 ...

  10. 浅谈:redis的主从复制 + 哨兵模式

    浅谈:redis的主从复制 + 哨兵模式 主从模式 ​ 在谈论redis的主从复制之前,我们先回想下mysql的主从搭建过程,第一步呢首先要在主库服务器中修改my.cnf,开启一下bin_log功能, ...