MyBatis注解实现动态SQL问题

 更新时间:2023年02月07日 08:58:39   作者:杨哥学编程  
这篇文章主要介绍了MyBatis注解实现动态SQL问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

MyBatis注解实现动态SQL

在 Mybatis 中,使用注解可以很方便的进行sql操作,但很多动态 SQL 都是由 xml 配置实现的。

而随着 SpringBoot的逐渐发展,越来越多的配置由配置文件转成注解的形式。其中包括动态 SQL 。

Mybatis 的注解中,使用太过长的sql语句看起来很不美观。

@Select("SELECT title from book_tbl where id = #{id}")

如果想要在 mapper 中进行非必要关键字的查询时就需要使用动态 SQL,与 xml 配置不同的是,@Select注解中 SQL 语句必须以 <script> 标签包裹。

@Select("<script>"+
			"SELECT id " +
            "from book_tbl" +
            "<where>" +
            "<if test 'company_id != null'>" +
            "and company_id = #{company_id}" +
            "</if>" +
            "<if test 'title != null'>" +
            "and title like CONCAT('%',#{title},'%')" +
            "</if>" +
            "</where>" +
            "ORDER BY create_time desc,rank desc" +
        "</script>")

MyBatis动态拼接 SQL参数

之前用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>

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 详解Java中wait和sleep的区别

    详解Java中wait和sleep的区别

    这篇文章主要介绍了Java中wait和sleep的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-03-03
  • SpringBoot3文件管理操作方法

    SpringBoot3文件管理操作方法

    这篇文章主要介绍了SpringBoot3文件管理,本文案例只围绕普通文件和Excel两种类型进行代码实现,包括工程搭建、上传下载操作,需要的朋友可以参考下
    2023-08-08
  • java中Servlet程序下载文件实例详解

    java中Servlet程序下载文件实例详解

    在本篇文章里小编给大家整理的是一篇关于java中Servlet程序下载文件实例内容,有兴趣的朋友们可以学习参考下。
    2021-02-02
  • Java Iterator迭代器与foreach循环代码解析

    Java Iterator迭代器与foreach循环代码解析

    这篇文章主要介绍了Java-Iterator迭代器与foreach循环,主要包括Iterator迭代器接口的操作方法和foreach 循环语法解析,需要的朋友可以参考下
    2022-04-04
  • Java_Spring之Spring 中的事务控制

    Java_Spring之Spring 中的事务控制

    这篇文章主要介绍了Java Spring中的事务控制,事务控制要明确内容,事务的控制都是基于AOP的,感兴趣的小伙伴可以参考阅读本文
    2023-04-04
  • Spring中BeanFactory和ApplicationContext的作用和区别(推荐)

    Spring中BeanFactory和ApplicationContext的作用和区别(推荐)

    这篇文章主要介绍了Spring中BeanFactory和ApplicationContext的作用和区别,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-09-09
  • Spring boot异步任务原理全面分析

    Spring boot异步任务原理全面分析

    这篇文章主要介绍了Spring boot异步任务原理,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-11-11
  • 一篇文章教你如何用Java自定义一个参数校验器

    一篇文章教你如何用Java自定义一个参数校验器

    这篇文章主要介绍了使用java自定义一个参数校验器,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习
    2021-09-09
  • Java8 Instant时间戳使用小记

    Java8 Instant时间戳使用小记

    这篇文章主要给大家介绍了关于Java8 Instant时间戳使用的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-01-01
  • SpringBoot中注册Bean的10种方式总结

    SpringBoot中注册Bean的10种方式总结

    在Spring Boot应用中,Bean是构成应用的核心组件,Spring容器负责管理这些Bean,包括它们的创建、配置、组装、管理和销毁,在Spring Boot中,有多种方式可以注册Bean,本文将详细介绍这些不同的注册方式,并给出相应的示例代码和适用场景,需要的朋友可以参考下
    2024-08-08

最新评论