【 动态SQL 的使⽤ 】

news2024/11/23 19:57:36

文章目录

  • 一、动态 SQL 是什么
  • 二、动 态 SQL 标签
    • 2.1 < if >标签
    • 2.2 < trim >标签
    • 2.3 < where >标签
    • 2.4 < set >标签
    • 2.5 < foreach >标签

一、动态 SQL 是什么

Mybatis 动态 sql 可以让我们在 Xml 映射文件内,以标签的形式编写动态 sql,完成逻辑判断和动态拼接 sql 的功能。动态 SQL 是 MyBatis 的强大特性之一,在 JDBC 或其它类似的框架中,开发人员通常需要手动拼接 SQL 语句。根据不同的条件拼接 SQL 语句是一件极其痛苦的工作。例如,拼接时要确保添加了必要的空格,还要注意去掉列表最后一个列名的逗号。而动态 SQL 恰好解决了这一问题,可以根据场景动态的构建查询

虽然动态 SQL 十分便利,但有时候在执行性能 (效率)上面不如静态SQL,而且使用不恰当,往往会在安全方面存在隐患 (SQL 注入式攻击)

二、动 态 SQL 标签

在这里插入图片描述

动 态 SQL 标签总共有上述9种,我们就最常用的举例说明 !!如下

2.1 < if >标签

在注册⽤户的时候,可能会有这样⼀个问题,如下图所示:

在这里插入图片描述

注册分为两种字段:必填字段和⾮必填字段,那如果在添加⽤户的时候有不确定的字段传⼊,程序应该如何实现呢?

⽐如添加头像 photo的时候为⾮必填字段,具体实现如下:

mapper 接口实现:

// 添加用户,添加用户时 photo是非必传参数
public int add2(UserInfo userInfo);

xml 文件实现:

    <!-- 添加用户,添加用户时 photo是非必传参数 -->
    <insert id="add2">
        insert into userinfo(username,password
        <if test="photo!=null">
            ,photo
        </if>
        ) values(#{username},#{password}
        <if test="photo!=null">
            ,#{photo}
        </if>
        )
    </insert>

测试代码:

    @Test
    void add2() {
        UserInfo userinfo = new UserInfo();
        userinfo.setUsername("李四");
        userinfo.setPassword("123");
        int result = userInfoMapper.add2(userinfo);
        log.info("添加用户的结果:" + result);
    }

我们在测试代码中并没有设置 photo 参数

查看运行结果:

在这里插入图片描述

查看数据库结果:

在这里插入图片描述

由此可见,我们的 if 标签起到了作用,那它的语法到底是什么意思呢?

在这里插入图片描述


2.2 < trim >标签

之前的插⼊⽤户功能,只是有⼀个 photo 字段可能是选填项,如果有多个字段,⼀般考虑使⽤
< trim >标签结合< if >标签,对多个字段都采取动态⽣成的⽅式

标签中有如下属性:

  1. prefix:表示整个语句块,以prefix的值作为前缀
  2. suffix:表示整个语句块,以suffix的值作为后缀
  3. prefixOverrides:表示整个语句块要去除掉的前缀
  4. suffixOverrides:表示整个语句块要去除掉的后缀

调整上述插入语句:

<insert id="add3">
        insert into userinfo
        <trim prefix="(" suffix=")" suffixOverrides=",">
            <if test="username!=null">
                username,
            </if>
            <if test="password!=null">
                password,
            </if>
            <if test="photo!=null">
                photo
            </if>
        </trim>
        values
        <trim prefix="(" suffix=")" suffixOverrides=",">
            <if test="username!=null">
                #{username},
            </if>
            <if test="password!=null">
                #{password},
            </if>
            <if test="photo!=null">
                #{photo}
            </if>
        </trim>
    </insert>

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

  1. 基于 prefix 配置,开始部分加上 (
  2. 基于 suffix 配置,结束部分加上 )
  3. 多个 组织的语句都以 , 结尾,在最后拼接好的字符串还会以 , 结尾,会基于
    suffixOverrides 配置去掉最后⼀个 ,
  4. 注意 <if test=“createTime != null”> 中的 createTime 是传⼊对象的属性

2.3 < where >标签

传⼊的⽤户对象,根据属性做where条件查询,⽤户对象中属性不为 null 的,都为查询条件。如
user.username 为 “a”,则查询条件为 where username=“a”

实现 mapper 接口

// <where> 标签 ! 传⼊的⽤户对象,根据属性做where条件查询,⽤户对象中属性不为 null 的,都为查询条件
public UserInfo getUserById1(@Param("id") Integer id,@Param("username") String username);

xml 文件实现

    <!-- 去除前面的 and -->
    <select id="getUserById1" resultMap= "BaseMap">
        select * from userinfo
        <where>
            <if test="id!=null">
                id=#{id}
            </if>
            <if test="username!=null">
                and username=#{username}
            </if>
        </where>
    </select>

测试代码:

    @Test
    void getUserById1() {
        UserInfo userInfo = userInfoMapper.getUserById1(null,"admin");
        log.info("chaxunxinxi:" + userInfo);
    }

此时我们没有传入id,观察运行结果

在这里插入图片描述

由此可见,where 后面只生成了一个条件 !!当也传入 id 参数再观察

在这里插入图片描述

此时where 后面有两个条件 !!

注意:

  1. 以上< where >标签也可以使⽤ < trim prefix=“where” prefixOverrides=“and” > 替换
  2. where 中所有的参数都为空时,where 的 sql 语句 就不会生成了
  3. where 标签可以去除最前面的 and 标签

2.4 < set >标签

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

mapper 接口实现:

int update2(UserInfo userInfo);

xml 文件实现:

    <!-- <set> 标签 -->
    <update id="update2">
        update userinfo
        <set>
            <if test="username!=null">
                username=#{username},
            </if>
            <if test="password!=null">
                password=#{password},
            </if>
            <if test="photo!=null">
                photo=#{photo}
            </if>
        </set>
        where id=#{id}
    </update>

测试代码:

    @Test
    void update2() {
        UserInfo userInfo = new UserInfo();
        userInfo.setId(45);
        userInfo.setUsername("haha");
        userInfo.setPassword("123");
        userInfo.setPhoto("default.png");
        int result = userInfoMapper.update2(userInfo);
        log.info("update 2 修改的结果为:" + result);
    }

修改前的数据表
在这里插入图片描述

修改后的数据表
在这里插入图片描述

我们尝试只传入 username 参数,效果如下:
在这里插入图片描述

在这里插入图片描述

可以看到,只有 username 字段被修改 ,而没有传入参数的字段还是保持原状 !!

注意:

  1. < set >标签可以自动去除最后一个逗号
  2. 同样可以使用 < trim > 标签代替 如下
    在这里插入图片描述

2.5 < foreach >标签

当我们需要根据多个用户 id 来删除用户数据时,平常做法如下:

delete from userinfo where id in ( ? , ? , ? )

我们会发现当删除的用户过多时, in() 里面工作量太大了,十分麻烦 !那是否有更简便的做法呢
答案是肯定的,我们可以思考将所有id 放在一个集合里面,然后直接遍历集合就行了,此时就可以考虑到 < foreach >标签了

对集合进⾏遍历时可以使⽤该标签,< foreach >标签有如下属性:

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

示例:根据多个用户 id 来删除用户数据

mapper 接口实现:

int delIds(List<Integer> ids);

xml 文件实现:

    <!-- <foreach> 标签 对集合进行遍历:根据多个id 删除 -->
    <delete id="delIds">
        delete from userinfo where id in
        <foreach collection="ids" open="(" close=")" item="id" separator=",">
            #{id}
        </foreach>
    </delete>

测试代码:

    @Test
    void delIds() {
        List<Integer> list = new ArrayList<>();
        list.add(43);
        list.add(44);
        list.add(45);
        int result = userInfoMapper.delIds(list);
        log.info("批量删除的结果:" + result);
    }

删除之前的数据表
在这里插入图片描述

运行结果:

在这里插入图片描述

删除后的数据表
在这里插入图片描述

结果显而易见,删除成功了 !!


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/457255.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【多微电网】含多微电网租赁共享储能的配电网博弈优化调度(Matlab代码实现)

&#x1f4a5; &#x1f4a5; &#x1f49e; &#x1f49e; 欢迎来到本博客 ❤️ ❤️ &#x1f4a5; &#x1f4a5; &#x1f3c6; 博主优势&#xff1a; &#x1f31e; &#x1f31e; &#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 …

计算机办公自动化——Python批量生成请假条

Python使用openpyxl、docx批量生成请假条 前言第三方库的安装示例代码运行效果 前言 加入你有一个下图所示的表格&#xff0c;需要批量生成他们的请假条&#xff0c;你会选择如何做呢&#xff1f;是一步一步的手打&#xff0c;还是呼唤请假人手打呢&#xff1f; 下面我们来看…

STM32学习,从点灯开始

大家好&#xff0c;我是程序员小哈。 综合实例&#xff1a;自动洗碗机的分享&#xff0c;上周五的直播完成了核心板的焊接&#xff0c;板子设计好了&#xff0c;也焊接完毕了&#xff0c;那么如何验证是否正确呢&#xff0c;既然是从0到1的教程&#xff0c;那么我们就先实现一…

PyTorch深度学习实战 | 基于深度学习的电影票房预测研究

基于深度学习的映前票房预测模型(Cross&Dense网络结构模型)&#xff0c;该模型通过影片基本信息如&#xff1a;电影类型、影片制式、档期和电影的主创阵容和IP特征等信息对上映影片的票房进行预测。 本篇采用451部电影作为训练模型&#xff0c;最后再在194部影片上进行测试…

Spring AOP实现原理

从入口 org.springframework.context.support.AbstractApplicationContext#refresh 开始看 找到Bean的创建方法进入: 再进入详细方法: 找到getBean(beanName)&#xff1a; 找到doGetBean(xxx,xxx,xxx,xxx); 找到实际的Bean创建方法createBean(beanName, mdb, args);可以非常明显…

【C++学习笔记】变量和基本类型

2.1 基本内置类型 C中包括 算数类型(arithmetic type) 和 空类型(void) 的数据类型&#xff0c;其中&#xff0c;算数类型包括字符、整型数、布尔值和浮点数&#xff1b;空类型不对应具体的值&#xff0c;当函数不返回值时用void作为返回类型 2.1.1算数类型 对于数组或者字符…

一篇搞定CDH 5.x 核心概念与集群部署

一、概述 1.1 Hadoop发行商 Apache:开源免费 CDH: Clouder公司研发。只支持64位操作系统。更加详细信息后面会介绍。 HDP: Hortonworks公司研发。 1.2 公司常用版本及介绍 apache -> cdh | hdp 常见问题&#xff1a; apache与cdh的比较&#xff1f;&#xf…

在国内怎么玩chatgpt,有可行的gpt游玩攻略么

首先你想玩chatgpt&#xff0c;你要明白一点这是一个国外的软件&#xff0c;所以你懂的&#xff0c;如果你不会魔法&#xff0c;那么就必须要改其他途径去探索游玩咯。今天我们就来探讨一下国内怎么玩chatgpt&#xff0c;可行的gpt游玩攻略。 一.Chatgpt的版本 我们先来认识一…

61 openEuler 22.03-LTS 搭建MySQL数据库服务器-管理数据库用户

文章目录 61 openEuler 22.03-LTS 搭建MySQL数据库服务器-管理数据库用户61.1 创建用户示例 61.2 查看用户示例 61.3 修改用户61.3.1 修改用户名61.3.2 修改用户示例61.3.3 修改用户密码61.3.4 修改用户密码示例 61.4 删除用户示例 61.5 用户授权示例 61.6 删除用户权限示例 61…

看完这篇文章你就彻底懂啦{保姆级讲解}-----(面试刷题链表相交) 2023.4.24

目录 前言面试题&#xff08;链表相交&#xff09;—&#xff08;保姆级别讲解&#xff09;分析题目&#xff1a;链表相交代码&#xff1a;算法思想 结束语 前言 本文章一部分内容参考于《代码随想录》----如有侵权请联系作者删除即可&#xff0c;撰写本文章主要目的在于记录自…

LVS负载均衡—DR模式

DR模式的特点 &#xff08;1&#xff09;Director Server&#xff08;调度器&#xff09; 和 Real Server&#xff08;节点服务器&#xff09; 必须在同一个物理网络中。 &#xff08;2&#xff09;Real Server可以使用私有地址&#xff0c;也可以使用公网地址。如果使用公网…

零代码平台如何帮助服装企业实现数字化转型?

随着互联网的不断发展&#xff0c;数字化转型已经成为各行各业必须跨越的一道坎&#xff0c;而服装行业也不例外。 但是&#xff0c;服装行业相对于其他行业来说&#xff0c;数字化转型面临着更多的挑战&#xff1a; 生产环节复杂&#xff1a;服装制造涉及到复杂的生产工序&a…

问题定位及解决方案

1.视频沉浸页快速滑动后&#xff0c;必现不能向下划动 复现步骤&#xff1a; 进入视频沉浸页&#xff0c;快速向下划动&#xff0c;滑动到第一页最后一个时&#xff0c;不能再向下划动。 解决步骤&#xff1a; 1.确定请求API&#xff1a; mtop.aliexpress.ugc.feed.video.lis…

Vivado关联第三方编辑器的方法

​Vivado是一个非常强大的工具&#xff0c;但是在一些方面可能不能完全满足我们的需求&#xff0c;比如代码编辑器的功能。幸运的是&#xff0c;Vivado允许我们关联第三方编辑器来扩展其代码编辑器的功能。在本文将介绍如何配置Vivado与第三方编辑器一起使用&#xff0c;并提供…

Pulsar幂等性开发的设计文档

PIP: https://github.com/apache/pulsar/issues/19744 具体设计 每个TC维护一个Map<ClientName,List> terminatedTxnMetaMap&#xff0c;维护每个客户端最新N个事务的状态&#xff0c;事务结束前&#xff0c;会把事务元数据写入这个List里&#xff0c;同时写入一个Compa…

高分辨率光学遥感图像水体分类综述2022.03

本文是Water body classification from high-resolution optical remote sensing imagery: Achievements and perspectives的学习笔记。 相关资源被作者整理到&#xff1a;这里 文章目录 Introduction基本知识 挑战和机遇挑战1. 有限的光谱信息和小场景覆盖2. 形状、大小和分布…

开放原子训练营(第三季)RT-Thread Nano学习营北京站

开放原子训练营&#xff08;第三季&#xff09;RT-Thread Nano学习营北京站学习心得 文章目录 开放原子训练营&#xff08;第三季&#xff09;RT-Thread Nano学习营北京站学习心得RT-Thread简介会议议程介绍RT-Thread Nano介绍RT-Thread Nano实操训练总结 RT-Thread简介 RT-Th…

【网络安全】XXE--XML外部实体注入

XXE XXE定义XML初识菜鸟xml概念初识DTD解答疑虑1&#xff1a;&#xff01;DOCTYPE是干什么用的疑虑2&#xff1a;&#xff01;ELEMENT是干什么用的疑虑3&#xff1a;#PCDATA是干什么用的疑虑4&#xff1a;为什么元素要再次声明类型 内部实体和外部实体的区别内部实体外部实体通…

【Spring篇】DI相关内容

&#x1f353;系列专栏:Spring系列专栏 &#x1f349;个人主页:个人主页 目录 一、setter注入 1.环境准备 2.注入引用数据类型 3.注入简单数据类型 二、构造器注入 1.环境准备 2.构造器注入引用数据类型 3.构造器注入多个引用数据类型 4.构造器注入多个简单数据类型 …

OAuth2学习(实操OAuth微信登录)

文章目录 前言1 OAuth2基本概念2 网站应用微信登录2.1 大概流程2.2 前期准备2.3 将微信登录二维码内嵌到自己页面2.3.1 后端接口编写(向前端提供参数)2.3.2 前端显示二维码页面 2.4 编写回调接口2.4.1 回调接口根据code获取access_token 这个令牌2.4.2 回调接口根据access_toke…