mybatis 动态拼接 sql参数

本文介绍如何在MyBatis中实现动态SQL拼接,包括使用foreach、set、choose、trim等标签处理不确定数量参数的方法,并附带相关示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

之前用JPA可以在@Query注解上拼接sql,实现动态查询;现在用mybatis,如何实现sql的动态拼接参数

举例,在JPA中可以实现类似于下面的sql拼接

plateNumber2 = plateNumber2 + "," + plateNumber;
String[] split = plateNumber2.split(",");
StringBuffer stringBuffer = new StringBuffer();
stringBuffer.append("SELECT plate_number from p_park_record where park_key = '"+parkDO.getParkKey()+"'");
stringBuffer.append(" and (");
for (int i = 0; i < split.length; i++) {
    stringBuffer.append(" plate_number = "+split[i]);
    if (i != split.length-1){
        stringBuffer.append(" OR ");
    }
}
stringBuffer.append(" )");
stringBuffer.append(" order by sensor_exit_time desc");
stringBuffer.append(" limit 1");

此代码中,核心问题是for循环,将不确定数量的同一个参数做拼接
那么在mybatis中该如何实现

mybatis实现:

plateNumber2 = plateNumber2 + "," + plateNumber;
ParkRecordDO parkRecordDO = new ParkRecordDO();
parkRecordDO.setPlateNumber(plateNumber2);
parkRecordDO.setParkKey(parkDO.getParkKey());
String onlyPlateNum = parkRecordMapper.getOnlyPlateNum(parkRecordDO);
    <select id="getOnlyPlateNum" parameterType="ParkRecordDO" resultType="java.lang.String">
        SELECT plate_number from p_park_record where park_key = #{parkKey} and sensor_exit_time is not null and
        <foreach collection="plateNumber.split(',')" index="index" item="id" open="(" separator=" or " close=")">
            plate_number = #{id}
        </foreach>
        order by sensor_exit_time desc
        limit 1
    </select>

如此实现,与JPA拼接效果相同,重点在于“<foreach”标签中separator属性,该属性为元素之间的分隔符,将分隔符设置为“or”,以此来实现该程序逻辑的拼接

最后补充几个知识点
foreach标签的使用

<!--
        foreach
        collection:指定要遍历的集合
                    list类型的参数会特殊处理封装在map中,map的key就叫list
              item:将当前遍历出的元素赋值给指定的变量
              separator:元素之间的分割符
              open: 遍历出的所有结果拼接一个开始字符
              close:遍历出的所有结果拼接一个结束字符
              index:索引,遍历list的时候是索引,item就是当前值;
                          遍历map时index标识的就是map的key,item就是map的值
              #{变量名}: 就能取出变量的值就是当前遍历出的元素
-->
<select id="queryUserByForEach" resultType="user">
    select * from easybuy_user where id in
    <foreach collection="ids" item="item_id" separator=","
             open="(" close=")">
        #{item_id}
    </foreach>
</select>


<insert id="batchSaveUser" parameterType="user">
    insert into easybuy_user(loginName, userName, password) values
    <foreach collection="userList" item="user" separator=",">
        (#{user.loginName}, #{user.userName}, #{user.password})
    </foreach>
</insert>

set标签的使用

<!--
        使用set标签或者trim标签与if标签相结合
        实现动态更新sql语句的拼接
-->
<update id="updateUserByCondition" parameterType="user">
    update easybuy_user
    <set>
        <if test="userName != null">
            username = #{userName},
        </if>
        <if test="email != null">
            email = #{email},
        </if>
    </set>
    where id = 26;
</update>

choose和when标签的使用(作用类似与java中的switch-case)

<select id="queryUserByChoose" resultType="com.unistart.entity.User">
    select * from easybuy_user
    <where>
        <choose>
            <when test="id != null">
                id=#{id}
            </when>
            <when test="userName != null">
                userName like #{userName}
            </when>
            <when test="sex != null">
                sex = #{sex}
            </when>
            <otherwise>
                1=1;
            </otherwise>
        </choose>
    </where>
</select>

trim标签的使用

<!--
        trim标签的使用:解决后面多出的and或or
        prefix="":前缀,trim标签体中是整个字符串拼串后的结果
                prefix给拼串后的整个字符串加一个前缀
        prefixOverrides="":前缀覆盖:去掉整个字符串前面多余的字符
        suffix="":后缀,给拼串后的整个字符串加一个后缀
        suffixOverrides="":后缀覆盖,去掉整个字符串后面多余的字符
-->
<select id="queryUserByTrim" resultType="user">
    select * from easybuy_user
    <trim prefix="where" suffixOverrides="and">
        <if test="id != null">
            id = #{id} and
        </if>
        <if test="userName != null and userName!=&quot;&quot;">
            userName like #{userName} and
        </if>
        <if test="email != null and email!=&quot;&quot;">
            email = #{email} and
        </if>
    </trim>
</select>

where标签的使用

<!--
        查询用户:要求,携带哪些字段查询条件就按这些字段进行查询
        使用的OGNL,类似与EL表达式

        从参数中取值进行判断,遇见特殊符号使用转义字符
        使用where标签时要注意,它只能去除第一个多出来的and或or
-->
<select id="queryUserByCondition" resultType="user">
    select * from easybuy_user
    <where>
        <if test="id != null">
            and id = #{id}
        </if>
        <if test="userName != null and userName!=&quot;&quot;">
            and userName like #{userName}
        </if>
        <if test="email != null and email!=&quot;&quot;">
            and email = #{email}
        </if>
    </where>
</select>
### MyBatis 动态 SQL 拼接 #### 使用 `if` 标签实现条件判断 为了在 MyBatis 中实现动态 SQL 语句的拼接,可以利用 `<if>` 标签来进行条件判断。当给定的条件成立时,则会将对应的 SQL 片段加入到最终执行的 SQL 语句中;反之则不会被包含进去。 ```xml <select id="findActiveBlogWithTitleLike" parameterType="map" resultType="Blog"> SELECT * FROM BLOG WHERE state = 'ACTIVE' <if test="title != null"> AND title like #{title} </if> </select> ``` 这段代码展示了如何基于传入参数的存在与否决定是否添加额外的过滤条件[^3]。 #### 利用 `foreach` 进行集合遍历 对于需要处理多个值的情况,比如 IN 查询或是批量更新/删除操作,MyBatis 提供了 `<foreach>` 标签用于迭代 Java 集合对象并生成相应的占位符或实际数值列表。 ```xml <delete id="batchDeleteUsersByIds" parameterType="list"> DELETE FROM users WHERE user_id IN <foreach item="id" index="index" collection="list" open="(" separator="," close=")"> #{id} </foreach> </delete> ``` 上述例子说明了怎样通过指定 `collection`, `item`, `index`(可选), `open`, `separator`, 及 `close` 属性来自动生成适合数据库使用的 IN 子句[^2]。 #### 组合多种标签完成复杂逻辑 除了单独使用某个特定功能外,在实际项目开发过程中往往还需要综合运用不同的 XML 元素来达到更精细控制的目的。下面是一个较为完整的案例: ```xml <select id="getPostsByCriteria" parameterType="PostSearchCriteria" resultType="Post"> SELECT p.* FROM posts p <where> <if test="categoryIds != null and categoryIds.size() > 0"> AND p.category_id IN <foreach item="categoryId" index="idx" collection="categoryIds" open="(" separator="," close=")"> #{categoryId} </foreach> </if> <if test="authorId != null"> AND author_id = #{authorId} </if> <!-- 更多条件... --> </where> </select> ``` 这里不仅有简单的字段匹配还有针对数组类型的特殊处理,并且借助于内置的 `<where>` 来自动管理第一个 AND/OR 关键字的位置问题[^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值