【Mybatis】动态 SQL

动态 SQL

动态 sql 是 Mybatis 的强⼤特性之⼀,能够完成不同条件下不同的 sql 拼接。

<if>标签

前端用户输入时有些选项是非必填的, 那么此时传到后端的参数是不确定的, 比如:

那如果在添加⽤户的时候有不确定的字段传⼊,程序应该如何实现呢?

这个时候就需要使⽤动态标签 来判断了,⽐如添加的时候性别 country 和 city 为⾮必填字段,具体实现如下:

javascript 复制代码
    <insert id="insert" parameterType="com.example.springboot3.model.User">
        insert into userinfo (
              username, phone, email,
              <if test="country != null">
                    country,
              </if>
              <if test="city != null">
                    city,
              </if>
        ) values (
              #{username},#{phone}, #{email},
              <if test="country != null">
                    #{country},
              </if>
              <if test="city != null">
                    #{city},
              </if>
        )
    </insert>

这样当 test 中判断结果为 true 时才会拼接上去.

<trim>标签

上面的插⼊⽤户功能,country 和 city 是选填项,如果所有字段都是选填项,就考虑使⽤ <trim> 标签结合 <if> 标签,对多个字段都采取动态⽣成的⽅式。

<trim>标签中有如下属性:

  • prefix:表示整个语句块,以prefix的值作为前缀
  • suffix:表示整个语句块,以suffix的值作为后缀
  • prefixOverrides:表示整个语句块要去除掉的前缀
  • suffixOverrides:表示整个语句块要去除掉的后缀
javascript 复制代码
    <insert id="insert" parameterType="com.example.springboot3.model.User">
        insert into userinfo
        <trim prefix="(" suffix=")" suffixOverrides=",">
            <if test="username != null">
                username,
            </if>
            <if test="phone != null">
                phone,
            </if>
            <if test="email != null">
                email,
            </if>
            <if test="country != null">
                country,
            </if>
            <if test="city != null">
                city,
            </if>
        </trim>
        <trim prefix="values (" suffix=")" suffixOverrides=",">
            <if test="username != null">
                #{username},
            </if>
            <if test="phone != null">
                #{phone},
            </if>
            <if test="email != null">
                #{email},
            </if>
            <if test="country != null">
                #{country},
            </if>
            <if test="city != null">
                #{city},
            </if>
        </trim>
    </insert>

在以上 sql 动态解析时,会对 <trim> 部分做如下处理:

  • 基于 prefix 配置,开始部分加上 (
  • 第二个 <trim> 开始部分会加上 values (
  • 基于 suffix 配置,结束部分加上 )
  • 多个 <if> 组织的语句都以 , 结尾,在最后拼接好的字符串还会以 , 结尾,会基于 suffixOverrides 配置去掉最后⼀个 ,

<where>标签

主要作用: 实现查询中的 where 替换, 可以实现无条件查询.

若存在查询条件会生成 where 的 sql 查询, 并且 where 标签可以去除最前面的 and

如果 where 中的所有参数为空, 那么 where 的 sql 就不会生成, 直接就是无条件查询

比如根据 国家(地区) 或者城市进行搜索, 根据属性做 where 条件查询,国家(地区) 或者城市不为 null 的,作为查询条件。

UserMapper 接口中的方法:

javascript 复制代码
List<User> selectByCondition(String country, String city);

UserMapper.xml 中的实现:

javascript 复制代码
    <select id="selectByCondition" resultType="com.example.springboot3.model.User">
        select * from userinfo
        <where>
            <if test="country != null">
                and country=#{country}
            </if>
            <if test="city != null">
                and city=#{city}
            </if>
        </where>
    </select>

以上<where>标签也可以使⽤ <trim prefix="where" prefixOverrides="and"> 替换。

<set>标签

主要作用: 进行修改操作时, 配合 if 处理非必须传输的参数, 特点是自动去除最后一个 ,

根据传⼊的⽤户对象属性来更新⽤户数据,可以使⽤标签来指定动态内容。

UserMapper 接口中的方法:

javascript 复制代码
int updateById(User user);

UserMapper.xml 中的实现:

javascript 复制代码
    <update id="updateById">
        update userinfo
        <set>
            <if test="username != null">
                username=#{username},
            </if>
            <if test="phone != null">
                phone=#{phone},
            </if>
            <if test="email != null">
                email=#{email},
            </if>
            <if test="country != null">
                country=#{country},
            </if>
            <if test="city != null">
                city=#{city},
            </if>
        </set>
        where id=#{id}
    </update>

<set>标签也可以使⽤ <trim prefix="set" suffixOverrides=","> 替换。

<foreach>标签

对集合进⾏遍历时可以使⽤该标签。

<foreach>标签有如下属性:

  • collection:绑定⽅法参数中的集合,如 List,Set,Map或数组对象
  • item:遍历时的每⼀个对象
  • open:语句块开头的字符串
  • close:语句块结束的字符串
  • separator:每次遍历之间间隔的字符串

比如根据用户 id 删除用户:

UserMapper 接口中的方法:

javascript 复制代码
int deleteByIds(List<Integer> ids);

UserMapper.xml 中的实现:

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

好啦! 以上就是对 Mybatis 动态 SQL 的讲解,希望能帮到你 !
评论区欢迎指正 !

相关推荐
IT成长日记21 分钟前
【Hive入门】Hive基础操作与SQL语法:DDL操作全面指南
hive·hadoop·sql·ddl操作
拿破轮1 小时前
查询Hologres或postgresql中的数据
数据库·postgresql
爱的叹息1 小时前
Spring和Spring Boot集成MyBatis的完整对比示例,包含从项目创建到测试的全流程代码
spring boot·spring·mybatis
声声codeGrandMaster2 小时前
django之账号管理功能
数据库·后端·python·django
Elastic 中国社区官方博客2 小时前
使用 LangGraph 和 Elasticsearch 构建强大的 RAG 工作流
大数据·数据库·人工智能·elasticsearch·搜索引擎·ai·全文检索
AscendKing2 小时前
mongo客户端操作mongodb记录
数据库·mongodb
千千寰宇3 小时前
[设计模式/Java] 设计模式之解释器模式【27】
数据库·设计模式
BXCQ_xuan3 小时前
Typecho博客网站头部SEO优化完整指南
运维·服务器·数据库·php·web
施嘉伟4 小时前
Oracle 11g RAC手动打补丁详细步骤
数据库·oracle
朴拙数科4 小时前
基于 RAG 的 Text2SQL 全过程的 Python 实现详解,结合 LangChain 框架实现自然语言到 SQL 的转换
python·sql·langchain