MyBatis 动态SQL的标签有哪些?如何使用?

目录

1. MyBatis 动态SQL标签有什么用?

2. if 标签

3. where 标签

4. trim 标签

5. choose,when,otherwise

6. foreach


1. MyBatis 动态SQL标签有什么用?

我来说一个场景大家就明白了,如下图,大家应该在铁路12306上买过火车票吧,可以看到,我们在选票的时候可以对车做很多种选择和筛选,系统会自动根据你的选择给你匹配适应的车次,其实它的底层就是采用的动态SQL拼接的方式完成的,系统会根据用户不同的选择动态生成不同的SQL语句,然后交给数据访问层去操作查询。

 下面我就来一个个介绍MyBatis框架给我们提供了哪些动态 SQL 拼接的标签。

2. if 标签

当我们实际开发多条件查询让用户选择的业务场景时,也许用户不会把所有筛选条件都选上,比如我们出去旅游,在美团上订酒店,可以按距离,按价格,按星级筛选,也许你不会把所有条件都选上,可能只选一个,但它也会查询出结果,这个时候它就可以采用 if 标签生成动态SQL语句达到我们的目的。

if 标签可以通过它提供的 test 属性对数据内容做判断,判断是否满足要求。若满足,if 标签内部的内容就会执行,反之则不执行。

使用方式就是

<if test = "判断条件">为true要执行的语句段<if>

示例如下代码所示

// 在 mapper 映射文件中写我们的 SELECT 查询语句
// 这里if标签内不能使用Java中的 && 连接符,mybatis 为我们提供了and关键字专门用来做连接操作
<select id="findByNameAndAge" resultType="User">
            select * from user where 1=1
            <if test="username != null and username != ''">
                and username = #{username}
            </if>
            <if test="age != null and age != ''">
                and age = #{age}
            </if>
        </select>
// 这里我只是写了 username 和 age 连个字段,实际开发过程中字段也许会很多,
// 这里举例就写的简单一些,其实都一样,当有其他字段时,继续追加 if 标签就可以了

如上举例 if 标签,我们可以在开发时,就可以将用户不一定选择的查询条件使用 if 标签包裹起来,test = "" 双引号内部就是我们的判断条件,不为空或者不为空字符串,和我们Java中的 if 判断非常相似,只有我们的 test 判断结果为 true,即用户输入的结果不为空也不是空字符串,说明用户输入的是有效数据,我们就将用户想要查询的过滤条件加入到查询条件中,如果为false,则该字段的筛选条件不参与SQL语句中。

这里有几个注意点我要提醒一下,各位同学千万注意!!!

(1)在 where 条件的后面,我们最好写一个恒成立的等式,例如1=1,2=2;

(2)每个 if 标签内的字段前面都要加上一个 and;

因为如果我们 if 标签内的结果都不成立,那么 SQL 语句就变成了 SELECT * FROM user WHERE 1=1。语法没有任何问题,但如果我们不加1=1 ,查询语句就会变成 SELECT * FROM user WHERR。WHERE 查询条件后面没有任何东西,就会出现语法级别的错误,这是绝对不允许出现的,为了避片此种情况,我们可以在 WHERE后面加上一个恒成立条件,使 WHERE 关键字无论何时都会起作用并且 SQL 语句语法不出错;

这是其中一点,另外一点就是,可以和我们 if 标签内的 and 完美连接。

我举一个例子,假设前端传递过来的数据 username 为空不满足条件,那么 SQL 语句就变成了 SELECT * FROM user WHERE 1=1 and age = #{age}。此时语法也没有任何问题,并且可以和 and 关键字巧妙连接。不但解决了可能出现的 SQL 语法问题,还解决了 and 关键字可能多余的情况。

3. where 标签

WHERE 标签的使用方式也很简单,大多是搭配 if 使用的。

使用方式 <WHERE>if 表标签内容</WHERE>

可以看到,在我刚才 if 标签中,我们将WHERE关键字写死了,不管后面的 if 标签是否生效,WHERE都会拼接到SQL语句中,这样的写法有些死板,而我们的 WHERE 标签就可以动态生成 WHERE 关键字。

将刚才 if 中的代码修改后如下:

<select id="findByNameAndAge" resultType="User">
            select * from user
            <where>
                <if test="username != null and username != ''">
                    username = #{username}
                </if>
                <if test="age != null and age != ''">
                    and age = #{age}
                </if>
            </where>
        </select>

我们只需要将所有的条件判断全部写到 WHERE 标签内部即可。

(1)WHERE标签内只要有一个条件生效,WHERE就会动态生成并拼接到SQL语句中;

(2)如果WHERE标签内部的都不执行,那么WHERE也不会生成;

(3)WHERE 还可以智能地将语句前面多余的 and 删除,也可以将 or 删除。

(4)and 或 or 连接符只能写在字段判断的前面,不能写在后面,否则 WHERE 标签将无法自动智能删除,只有写在前面才有效果;

例如,我的 username =#{username}不执行,那么理论上SQL语句就会变成 SELECT * FROM user WHERE and age = #{age}。但实际上,WHERE标签会智能将 and 删除,就会变成 SELECT * FROM user WHERE age = #{age}。

4. trim 标签

trim 标签是一个单独的标签,它可以单独使用,如下图,可以看到,该标签一共提供了四个属性

其中 prefix 翻译过来是"前缀",prefixOverrides 为 "前缀重写",suffix 为 "后缀",suffixOverrides 为 "后缀重写"。

prtefix/suffix:在trim标签中内容前面/后面添加指定内容;

prefixOverrides/suffixOverrides:在trim标签中内容前面/后面去掉指定内容;

而且 trim 标签中有内容时,trim就会生效,如果没有内容,trim标签也不会生效。

我们想要达到上面 WHERE 标签一样的效果,可以像下面这样写,使用 prefix 属性添加前缀 WHERE,使用 prefixOverrides 去掉多余的 and和or,他们中间用 "|" 连接。

<select id="findByNameAndAge" resultType="User">
            select * from user
            <trim prefix="where" prefixOverrides="and|or">
                <if test="username != null and username != ''">
                    username = #{username}
                </if>
                <if test="age != null and age != ''">
                    and age = #{age}
                </if>
            </trim>
        </select>

 当我们if 标签内 and 写在后面的时候,只需要使用 suffixOverrides 就可以达到效果,如下代码所示

<select id="findByNameAndAge" resultType="User">
            select * from user
            <trim prefix="where" suffixOverrides="and|or">
                <if test="username != null and username != ''">
                    username = #{username} and
                </if>
                <if test="age != null and age != ''">
                    age = #{age}
                </if>
            </trim>
        </select>

我们只需要将 prefixOverrides 改为 suffixOverrides就可以达到效果。 

5. choose,when,otherwise

这三个标签一般情况下会一起出现,它与我们Java中 if...else if...else 的功能非常相似。

使用方式如下

这里要注意,<when>标签至少需要写一个,而<othersise>最多只能写一个。

<choose>
    <when test="判断条件">...</when>
    <when test="判断条件">...</when>
    <otherwise>判断条件</otherwise>
</choose>

当我们写这样的一个标签体时,只要有一个条件成立,它就不会再继续向下判断了,直接将满足条件的字段拼接到SQL语句中。也就是说,该标签体内就算有多个条件都满足,但还是只会执行第一个满足条件的字段查询。

它们也可以搭配<WHERE></WHERE> 或者<trim></trim> 使用,大家可以自由选择,这里我举例使用<WHERE></WHERE>,示例代码举例如下

<select id="findByNameAndAge" resultType="User">
            select * from user
            <where>
                <choose>
                    <when test="username != null and username != ''">
                        username = #{username}
                    </when>
                    <when test="age != null and age != ''">
                        age = #{age}
                    </when>
                    <otherwise>uid = 1</otherwise>
                </choose>
            </where>
        </select>

当我们像上面这样写的时候,username 和 age 的判断只会执行一个,即使用户填写的都是有效数据,我们也会只执行第一个 username 的判断,所以在开发时需要根据具体的业务场来使用该标签体。如果都不满足,我们可以在 otherwise 中编写一个候选查询条件。

6. foreach

foreach 听名字我们也大概可以猜到它是用来循环的,使用该标签可以实现我们对数据的批量添加和批量删除操作。

foreach标签也为我们提供了几个属性,下面我会依次说明这些属性是做什么用的

cllection:代表要遍历的数组,Java接口在定义是最好使用@Parm进行标注;

item:表示数组中的每个元素;

open:表示该标签体以什么开头;

close :表示该标签体以什么结尾;

separator:表示数据之间的分隔符是什么;

现在有一个场景,根据传输过来的数组中的用户id批量删除这些用户信息。

就此场景,我们就可以使用<foreach>标签来完成;

代码如下

<delete id="deleteListsByUid">
        delete from user where uid in
        <foreach collection="userList" item="uid" open="(" close=")" separator=",">
            #{uid}
        </foreach>
    </delete>

我们批量数据操作时,前面有数据库关键字 in ,所以需要加"()",因此则是以 "(" 开头,以 ")" 结尾,所以标签属性 open = "(",close = ")"。其实 open 和 close也不常用,我们可以自己添加,这里只是给各位说明它们两个的作用。

此外,因为数组中有多个数据,因此每个数据在SQL语句中需要使用逗号隔开,因此 separator = ","。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值