MyBatis 中的动态 SQL 的相关使用方法

为什么会有动态SQL,把SQL写死不是比较方便吗?其实有很多的举例,这里我那一个常见的来说,像我们用户注册,会有必填字段和非必填字段,有些传来的参数不一样,那对应的SQL也不一样,因此,在这些的场景底下我们需要使用动态SQL来完成。

这里动态SQL我是使用XML的办法进行说明,因为使用注解不方便,而且不好观察哪里有误。这里我会先从一些常用的标签的介绍来对动态SQL进行使用的讲解。

1.1 <if>标签

XML 复制代码
    <insert id="insertByCondition">
        insert into userinfo(
            <if test="username != null">
                username
            </if>
            <if test="password != null">
                ,password
            </if>
            <if test="age != null">
                ,age
            </if>
            <if test="gender != null">
                ,gender
            </if>
        ) values (
            <if test="username != null">
                #{username}
            </if>
            <if test="password != null">
                ,#{password}
            </if>
            <if test="age != null">
                ,#{age}
            </if>
            <if test="gender != null">
                ,#{gender}
            </if>
        )
    </insert>

从上述代码可以简单看出来,<if> 标签是一个<if test = "xxxx">xxxx</if>的结构,然后另一个是运行后的结果,这里发现如果if里面都有值的话,可以完好的插入,但是它就一定没问题吗?下面一个图我就简单注释掉几个来,实验一下。

从上图可以简单的看错来,它是一个BadSql因为我们注掉了username导致,多了一个逗号那怎么样才能避免上述情况呢?那就要引入新的注解。

1.2<trim> 标签

这里我们先简单介绍一下<trim>标签里面几个常见的属性:

  1. prefix:表示整个语句块,以prefix的值作为前缀。

  2. suffix:表示整个语句块,以suffix的值作为后缀。

  3. prefixOverrides:表示整个语句块要去除掉前缀。

  4. suffixOverrides:表示整个语句块要去掉后缀。

XML 复制代码
<insert id="insertByCondition">
        insert into userinfo
        <trim prefix="(" suffix=")" prefixOverrides=",">
            <if test="username != null">
                username
            </if>
            <if test="password != null">
                ,password
            </if>
            <if test="gender != null">
                ,gender
            </if>
            <if test="age != null">
                ,age
            </if>
        </trim>
        values
        <trim prefix="(" suffix=")" prefixOverrides=",">
            <if test="username != null">
                #{username}
            </if>
            <if test="password != null">
                ,#{password}
            </if>
            <if test="gender != null">
                , #{gender}
            </if>
            <if test="age != null">
                ,#{age}
            </if>

        </trim>
    </insert>

上述因为username是不能为空,因此我把注掉username改为注掉gender,可以发现在<if>标签里面的问题给解决了。

1.3 <where>标签

XML 复制代码
<select id="queryUserByCondition" resultType="com.jincheng.mybatisdemo.model.UserInfo">
        select * from userinfo
        <where>
            <if test="age != null">
                age = #{age}
            </if>
            <if test="gender != null">
                and gender = #{gender}
            </if>
            <if test="deleteFlag">
                and delete_flag = #{deleteFlag}
            </if>
        </where>
    </select>
XML 复制代码
    <select id="queryUserByCondition" resultType="com.jincheng.mybatisdemo.model.UserInfo">
        select * from userinfo
        <trim prefix="where" prefixOverrides="and">
            <if test="age != null">
                age = #{age}
            </if>
            <if test="gender != null">
                and gender = #{gender}
            </if>
            <if test="deleteFlag">
                and delete_flag = #{deleteFlag}
            </if>
        </trim>
    </select>

通过上面可以发现第一个是使用<where>标签进行写的,而另一个是通过使用<trim>标签进行写的。可以简单比对一下<where>标签,生成where关键字,并且去除最前面的and 或者 or,如果where标签代码块,没有一个查询条件,会省略掉where关键字。也可以使用<trim prefix="where" prefixOverrides="and">替换,但是这种情况下,子元素没有内容时,where关键字也会保留。

1.4 <set>标签

XML 复制代码
<update id="updateByCondition">
        update userinfo
        <set>
            <if test="password != null">
                password = #{password},
            </if>
            <if test="age != null">
                age = #{age},
            </if>
            <if test="gender != null">
                gender = #{gender}
            </if>
        </set>
            where id = #{id}
    </update>

这里<set>标签,生成set关键字,去掉整个代码块后面的逗号。如果使用<trim prefix="set" suffixOverrides=",">可以进行替换。

1.5 <foreach>标签

简单来说一下<foreach>标签里面的属性:

  1. collection:绑定方法参数中的集合,如LIst,Set,Map或数组对象

  2. item:遍历是的每一个对象

  3. open:语句块开头的字符串

  4. close:语句块结束的字符串

  5. separator:每次遍历之间间隔的字符串。

一般用在批量删除中等等。

XML 复制代码
    <delete id="batchDelete">
        delete  from userinfo
        where id in
        <foreach collection="ids" open="(" close=")" item="id" separator=",">
            #{id}
        </foreach>
    </delete>

1.6 <inlcude>标签

这个标签用于在xml映射文件中配置的SQL,有时可能会存在很多重复的片段,此时就会在很多多余的代码。

这里我们需要对重复的代码片段进行抽取,将其通过<sql>标签封装到一个SQL片段,然后通过<include>标签进行引用:

1.<sql> : 定义可重用的SQL片段

  1. <include>:通过属性refid,指定包含的SQL片段。
XML 复制代码
    <sql id="selectAll">
        select * from userinfo
    </sql>

    <select id="select" resultType="com.jincheng.mybatisdemo.model.UserInfo">
        <include refid="selectAll"></include>
    </select>
相关推荐
不羁。。40 分钟前
【撸靶笔记】第八关:GET - Blind - Boolian Based - Single Quotes
数据库·sql·mybatis
西红柿维生素8 小时前
MyBatis SqlCommand+MethodSignature源码探究
mybatis
HeyZoeHey2 天前
Mybatis执行sql流程(一)
java·sql·mybatis
青川入梦2 天前
MyBatis极速通关上篇:Spring Boot环境搭建+用户管理实战
java·开发语言·mybatis
33255_40857_280592 天前
掌握分页艺术:MyBatis与MyBatis-Plus实战指南(10年Java亲授)
java·mybatis
勿在浮沙筑高台3 天前
无法获取实体类com.example.springdemo2.entity.po.UserPO对应的表名!
java·spring boot·mybatis
柯南二号3 天前
【Java后端】MyBatis-Plus 原理解析
java·开发语言·mybatis
Easocen3 天前
Mybatis学习笔记(五)
笔记·学习·mybatis
qq_三哥啊4 天前
【IDEA】设置Debug调试时调试器不进入特定类(Spring框架、Mybatis框架)
spring·intellij-idea·mybatis
柯南二号4 天前
【Java后端】Spring Boot 集成 MyBatis-Plus 全攻略
java·spring boot·mybatis