Mybatis-动态SQL

news2025/1/8 5:52:30

1.什么是动态SQL?        

Mabits是一个Java持久化框架,它提供了动态SQL的功能。动态SQL是一种根据不同条件动态生成SQL语句的技术。在Mabits中,动态SQL通常是通过使用一组特殊的标签和代码块来实现的,这些标签和代码块可以根据条件包含或排除不同的部分,从而生成不同的SQL语句。动态SQL可以让开发者构建更灵活、高效的数据库操作语句,因为他们可以根据实际需要创建出更加精确和优化的SQL语句。Mabits的动态SQL还允许开发者使用参数化查询,可以防止SQL注入攻击,并增强程序的安全性

接下来学习经常使用的几个标签

2.<if>标签

<if>标签是Mabits动态SQL中最常用的一个标签之一,它的作用是根据给定条件包含或排除不同的部分,以生成不同的SQL语句。在XML文件中,<if>标签通常被嵌套在其他标签内,如<select>、<insert>和<update>等标签内,用于控制生成的SQL语句的结构和内容。

<if>标签通常包含一个test属性,该属性被用于指定条件表达式。如果表达式的结果为true,则<if>标签内的内容会被包含在生成的SQL语句中;否则,这些内容会被忽略。以下代码段展示了如何在一个<select>标签内使用<if>标签来动态生成SQL语句:

常见场景:非必传参数时使用<if>

int addUser2(UserEntity user);

@Test
    void addUser2() {
        String username = "zhaoliu";
        String password = "123456";
        UserEntity user = new UserEntity();
        user.setUsername(username);
        user.setPwd(password);
        int result = userMapper.addUser2(user);
        System.out.println("修改行数: "+result);
    }
<insert id="addUser2">
        insert into userinfo(username,password
            <if test="photo != null">
                ,photo
            </if>
        ) values(#{username},#{pwd}
            <if test="photo != null">
                ,#{photo}
            </if>
        )
    </insert>

当不传入photo时,执行结果

JDBC Connection [HikariProxyConnection@579590740 wrapping com.mysql.cj.jdbc.ConnectionImpl@75b6dd5b] will not be managed by Spring
==>  Preparing: insert into userinfo(username,password ) values(?,? )
==> Parameters: zhaoliu(String), 123456(String)
<==    Updates: 1
Closing non transactional SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@380e1909]
修改行数: 1

传入photo时,执行结果

JDBC Connection [HikariProxyConnection@645717550 wrapping com.mysql.cj.jdbc.ConnectionImpl@609e57da] will not be managed by Spring
==>  Preparing: insert into userinfo(username,password ,photo ) values(?,? ,? )
==> Parameters: zhaoliu2(String), 123456(String), dog.png(String)
<==    Updates: 1
Closing non transactional SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@24d61e4]
修改行数: 1


//数据库
mysql> select*from userinfo;
+----+----------+----------+---------+---------------------+---------------------+-------+
| id | username | password | photo   | createtime          | updatetime          | state |
+----+----------+----------+---------+---------------------+---------------------+-------+
|  1 | lisi     | 123456   |         | 2022-12-06 17:10:48 | 2022-12-06 18:10:48 |     1 |
|  3 | zhangsan | 123456   |         | 2023-05-18 17:21:49 | 2023-05-18 17:21:49 |     1 |
|  4 | wangwu   | 123456   |         | 2023-05-18 17:36:28 | 2023-05-18 17:36:28 |     1 |
|  5 | zhaoliu  | 123456   |         | 2023-05-20 09:22:08 | 2023-05-20 09:22:08 |     1 |
|  6 | zhaoliu2 | 123456   | dog.png | 2023-05-20 09:35:06 | 2023-05-20 09:35:06 |     1 |
+----+----------+----------+---------+---------------------+---------------------+-------+
5 rows in set (0.00 sec)

 注意理解:

3.<trim>标签

学习<trim>标签之前先看一个问题

<insert id="addUser3">
        insert into userinfo(
        <if test="username != null">
            username,
        </if>
        <if test="pwd != null">
            password,
        </if>
        <if test="photo != null">
            photo
        </if>
        ) values(
        <if test="username != null">
            #{username},
        </if>
        <if test="pwd != null">
            #{pwd},
        </if>
        <if test="photo != null">
            #{photo}
        </if>
        )
    </insert>

当遇到这种所有参数都为非必填参数时,如果第一个参数传了值,后续都没传,就会出现(username,)这个问题,此时很明显SQL 语句是错误的,肯定不能成功执行或者其它类似这种问题

<trim>标签就能解决此类问题

<trim>标签也是Mabits动态SQL中常用的一个标签,主要用于快速生成包含WHERE、SET等关键字的SQL语句,同时还能够自动处理SQL语句中的逗号(,)和AND/OR等连接符。

<trim>标签通常包含以下属性:

  1. prefixOverrides:表示在生成SQL语句前需要忽略的字符前缀。
  2. suffixOverrides:表示在生成SQL语句后需要忽略的字符后缀。
  3. prefix:表示在生成SQL语句前需要添加的字符串。
  4. suffix:表示在生成SQL语句后需要添加的字符串。
  5. suffixOverrides:表示在生成SQL语句中需要忽略掉的字符串。

 用法一:删除前缀后缀

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

    </insert>

单元测试

@Transactional
    @Test
    void addUser3() {
        String username = "zhaoliu3";
        String password = "123456";
        UserEntity user = new UserEntity();
        user.setUsername(username);
        user.setPwd(password);
        int result = userMapper.addUser2(user);
        System.out.println("修改行数: "+result);
    }

当不传photo时,上一个pwd后的,被去除了

JDBC Connection [HikariProxyConnection@1228603887 wrapping com.mysql.cj.jdbc.ConnectionImpl@36c2d629] will be managed by Spring
==>  Preparing: insert into userinfo(username,password ) values(?,? )
==> Parameters: zhaoliu3(String), 123456(String)
<==    Updates: 1

用法二,解决多个非必传参数问题

    List<ArticleInfoVO> getListByIdOrTitle(@Param("id")Integer id ,@Param("title")String title);


<select id="getListByIdOrTitle" resultType="com.example.demo.entity.vo.ArticleInfoVO">
        select*from articleinfo
        <trim prefix="where" suffixOverrides="and">
            <if test="id!=null and id >0">
                id = #{id} and
            </if>
            <if test="title!=null and title!=''">
                title like concat('%',#{title},'%')
            </if>
        </trim>
    </select>


@Test
    void getListByIdOrTitle() {
        List<ArticleInfoVO> list = articleMapper.getListByIdOrTitle(null,null);
        System.out.println(list.size());
    }

执行结果:

传参都为空时,也就是trim中没有内容,那么prefix中的where也不会生产。就不用再使用

where 1=1 and ...的方式解决传参为空的问题了 

 传一个参数,前缀where就会执行

4.<where>标签

能更好的解决上述多个非必传参数问题

过使用<where>标签和<if>标签,可以根据用户传入的参数动态生成SQL语句的WHERE子句。如果参数不为空,则会包含相应的查询条件;否则,该查询条件会被忽略。

需要注意的是,由于<where>标签会自动处理WHERE子句中的AND和OR关键字,因此在使用时应该合理安排条件语句的位置,避免出现意外结果。

执行

 非常方便

不传id 传title

将and前缀清除了

注意:<where>只能去除sql语句前缀的关键字,不能去除后缀关键字

 <where>标签也可以用<trim prefix="where", prefixOverrides="and">替换

5.<set>标签

通过使用<set>标签和<if>标签,可以根据用户传入的参数动态生成SQL语句的SET子句。如果参数不为空,则会包含相应的更新内容;否则,该更新内容会被忽略

需要注意的是,由于<set>标签会自动处理SET子句中的逗号(,),因此在使用时应该合理安排更新内容的位置,避免出现意外结果

<set>标签通过将SET子句中的所有条件用逗号(,)连接起来,可以根据不同的更新情况动态生成满足需求的SQL语句

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

示例:修改lisi->lis2,password->456789

mysql> select*from userinfo;
+----+----------+----------+---------+---------------------+---------------------+-------+
| id | username | password | photo   | createtime          | updatetime          | state |
+----+----------+----------+---------+---------------------+---------------------+-------+
|  1 | lisi     | 123456   |         | 2022-12-06 17:10:48 | 2022-12-06 18:10:48 |     1 |
|  3 | zhangsan | 123456   |         | 2023-05-18 17:21:49 | 2023-05-18 17:21:49 |     1 |
|  4 | wangwu   | 123456   |         | 2023-05-18 17:36:28 | 2023-05-18 17:36:28 |     1 |
|  5 | zhaoliu  | 123456   |         | 2023-05-20 09:22:08 | 2023-05-20 09:22:08 |     1 |
|  6 | zhaoliu2 | 123456   | dog.png | 2023-05-20 09:35:06 | 2023-05-20 09:35:06 |     1 |
+----+----------+----------+---------+---------------------+---------------------+-------+

代码

//修改
    int updateUser(UserEntity user);

<update id="updateUser">
        update userinfo
        <set>
            <if test="username!=null and username!=''">
                username = #{username},
            </if>
            <if test="pwd!=null and pwd!=''">
                password = #{pwd},
            </if>
        </set>
        where id = #{id}
    </update>

    @Test
    void updateUser() {
        UserEntity user = new UserEntity();
        user.setUsername("lisi2");
        user.setPwd("456789");
        user.setId(1);
        int result = userMapper.updateUser(user);
        System.out.println("修改后: "+result);
    }

执行结果

JDBC Connection [HikariProxyConnection@236002428 wrapping com.mysql.cj.jdbc.ConnectionImpl@4d0e54e0] will not be managed by Spring
==>  Preparing: update userinfo SET username = ?, password = ? where id = ?
==> Parameters: lisi2(String), 456789(String), 1(Integer)
<==    Updates: 1
Closing non transactional SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@95eb320]
修改后: 1
mysql> select*from userinfo;
+----+----------+----------+---------+---------------------+---------------------+-------+
| id | username | password | photo   | createtime          | updatetime          | state |
+----+----------+----------+---------+---------------------+---------------------+-------+
|  1 | lisi2    | 456789   |         | 2022-12-06 17:10:48 | 2022-12-06 18:10:48 |     1 |
|  3 | zhangsan | 123456   |         | 2023-05-18 17:21:49 | 2023-05-18 17:21:49 |     1 |
|  4 | wangwu   | 123456   |         | 2023-05-18 17:36:28 | 2023-05-18 17:36:28 |     1 |
|  5 | zhaoliu  | 123456   |         | 2023-05-20 09:22:08 | 2023-05-20 09:22:08 |     1 |
|  6 | zhaoliu2 | 123456   | dog.png | 2023-05-20 09:35:06 | 2023-05-20 09:35:06 |     1 |
+----+----------+----------+---------+---------------------+---------------------+-------+

不传参数pwd时 

6.<foreach>标签

<foreach> 标签是 MyBatis 中的一个迭代标签,可以对集合对象进行遍历,并生成多条 SQL 语句(如 INSERTUPDATEDELETE 等)。

使用 <foreach> 标签,我们可以将一个集合对象的元素依次取出,作为 SQL 语句中的参数进行插入、更新或删除操作。常用的语法如下:

<foreach collection="collection" item="item" separator="separator" open="open" close="close"> 
    <!-- SQL 语句 --> 
</foreach>

其中,各个属性和元素的含义如下:

  • collection:指定要遍历的集合对象的属性名。
  • item:指定在遍历过程中每个元素所对应的变量名。
  • separator:指定在生成多条 SQL 语句时,不同语句之间的分隔符,默认为英文逗号。
  • open:指定生成的 SQL 语句的头部。
  • close:指定生成的 SQL 语句的尾部。

示例:删除文章表的数据

mysql> select*from articleinfo;
+----+-------+-----------+---------------------+---------------------+-----+--------+-------+
| id | title | content   | createtime          | updatetime          | uid | rcount | state |
+----+-------+-----------+---------------------+---------------------+-----+--------+-------+
|  1 | Java  | Java正文  | 2023-05-15 09:12:59 | 2023-05-15 09:12:59 |   1 |      1 |     1 |
|  2 | mysql | mysql正文 | 2023-05-19 11:14:49 | 2023-05-19 11:14:49 |   1 |      1 |     1 |
+----+-------+-----------+---------------------+---------------------+-----+--------+-------+
2 rows in set (0.00 sec)

此处进行回滚,不真的删除

代码:

//根据id批量删除文章
    int delByIdList(List<Integer> idList);

<delete id="delByIdList">
        <!--  delete from articleinfo where id in(1,2,3....)      -->
        delete from articleinfo
        where id in(
            <foreach collection="idList" item="id" separator=",">
                #{id}
            </foreach>
        )
    </delete>


@Transactional
    @Test
    void delByIdList() {
        List<Integer> idList = new ArrayList<>();
        idList.add(1);
        idList.add(2);
        idList.add(3);
//只有两条
        int result = articleMapper.delByIdList(idList);
        System.out.println("删除: "+result);
    }

 执行单元测试

JDBC Connection [HikariProxyConnection@918738473 wrapping com.mysql.cj.jdbc.ConnectionImpl@55fee662] will be managed by Spring
==>  Preparing: delete from articleinfo where id in( ? , ? , ? )
==> Parameters: 1(Integer), 2(Integer), 3(Integer)
<==    Updates: 2
Releasing transactional SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@4b960b5b]
删除: 2

上述代码没有使用open,close属性.将()写在了foreach标签外,这里加上

和刚才的写在标签外的方式作用是相同的

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

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

相关文章

性能测试——jmeter实时压测结果收集

这里写目录标题 前言一、压测监控平台组成二、性能监控平台部署 - InfluxDB三、性能监控平台部署 - JMeter四、性能监控平台部署 - Grafana五、性能监控平台部署 – 运行与结果展示 前言 测试报告 .vs. 压测监控 JMeter原生测试报告带来的“痛苦” • 不具备实时性 • 报告中的…

单体项目偶遇并发漏洞!短短一夜时间竟让老板蒸发197.83元

事先声明&#xff1a;以下故事基于真实事件而改编&#xff0c;如有雷同&#xff0c;纯属巧合~ 眼下这位正襟危坐的男子&#xff0c;名为小竹&#xff0c;他正是本次事件的主人公&#xff0c;也即将成为熊猫集团的被告&#xff0c;嗯&#xff1f;这究竟怎么一回事&#xff1f;欲…

通过白噪声的频谱处理产生任意光谱斜率(f^a)噪声(Matlab代码实现)

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

首站中科院!百度商业AI技术创新大赛开启巡回宣讲

近日&#xff0c;百度商业AI技术创新大赛正式启动&#xff0c;并于5月18日起开启高校巡回宣讲。 宣讲会首站落地中国科学院大学&#xff0c;中国科学院大学人工智能学院副院长、教授、博士生导师肖俊教授&#xff0c;百度商业研发主任架构师焦学武&#xff0c;百度商业资深工程…

springboot 集成 Swagger3(速通)

→ springboot 集成 Swagger2 ← 目录 1. 案例2. info 配置3. Docket 配置1. 开关配置2. 扫描路径3. 路径匹配4. 分组管理 4. 常用注解1. 说明2. 案例 1. 案例 这次直接使用 2.5.6 的 spring-boot 。 依赖&#xff1a; <parent><groupId>org.springframework.…

亏损?盈利?禾赛科技Q1财报背后的激光雷达赛道「现实」

随着禾赛科技在去年登陆美股&#xff0c;作为全球为数不多已经开始前装量产交付的激光雷达上市公司&#xff0c;财务数据的变化&#xff0c;也在一定程度上反映了行业的真实状况。 根据禾赛科技最新发布的今年一季度财报显示&#xff0c;公司季度净营收为4.3亿元&#xff08;人…

基于html+css的图展示91

准备项目 项目开发工具 Visual Studio Code 1.44.2 版本: 1.44.2 提交: ff915844119ce9485abfe8aa9076ec76b5300ddd 日期: 2020-04-16T16:36:23.138Z Electron: 7.1.11 Chrome: 78.0.3904.130 Node.js: 12.8.1 V8: 7.8.279.23-electron.0 OS: Windows_NT x64 10.0.19044 项目…

电表及配电监控系统的智能化发展

安科瑞虞佳豪 在电力领域&#xff0c;AI渗透率近年来也持续提升。今年3月&#xff0c;国家能源局发布《关于加快推进能源数字化智能化发展的若干意见》&#xff0c;文件中针对电力、、油气等行业数字化智能化转型&#xff0c;明确了指导思想和基本原则&#xff0c;从加快行业转…

Unity之ShaderGraph数据类型

前言 ShaderGraph是Unity引擎中的一个可视化着色器编辑器&#xff0c;它允许开发者使用节点和连接线的方式来创建自定义的着色器。使用ShaderGraph&#xff0c;开发者可以通过简单的拽和连接节点来创建复杂的着色器效果&#xff0c;而无需编写任何代码。 为了更好的学习Shader…

chatgpt赋能python:PythonWOL:简化计算机远程唤醒的一种方法

Python WOL&#xff1a;简化计算机远程唤醒的一种方法 随着人们越来越频繁地远程连接和控制计算机&#xff0c;计算机的远程唤醒功能变得越来越重要。WOL&#xff0c;即“Wake on LAN”&#xff0c;是一种使用网络信号远程唤醒计算机的技术。 在许多情况下&#xff0c;使用WO…

打家劫舍问题 Python题解

✅作者简介&#xff1a;人工智能专业本科在读&#xff0c;喜欢计算机与编程&#xff0c;写博客记录自己的学习历程。 &#x1f34e;个人主页&#xff1a;小嗷犬的个人主页 &#x1f34a;个人网站&#xff1a;小嗷犬的技术小站 &#x1f96d;个人信条&#xff1a;为天地立心&…

python 网络编程和http协议--网络编程,HTTP协议,Web服务器

一.网络编程 1.IP地址 给网络中的每一台设备进行编号. IPV4 IPV6 2.端口和端口号 端口的作用就是给运行的应用程序提供传输数据的通道。 端口号的作用是用来区分和管理不同端口的&#xff0c;通过端口号能找到唯一个的一个端口。 3.TCP协议 协议: 双方的约定. 网络传输协…

基于SpringBoot+微信小程序的点餐系统

✌全网粉丝20W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取项目下载方式&#x1f345; 一、项目背景介绍&#xff1a; 小程序外卖扫码点餐为…

【MQTT】关于部署含有MQTT协议的程序pod到K8S中出现的问题

1、如何在go-zero整合MQTT协议 整合EMQX与eclipse/paho.mqtt.golang实现TLS&#xff08;ssl&#xff09;单向认证 订阅、发布、解压缩gzip、zlib数据报文 https://ctraplatform.blog.csdn.net/article/details/130525974 1.1 、通过钩子函数一直出现Lost问题排查 场景&#xff…

人类睡眠EEG分析:附代码实现的方法学入门

导读 近年来&#xff0c;人类睡眠脑电图(EEG)研究激增&#xff0c;采用了越来越复杂的分析策略将电生理活动与认知和疾病联系起来。然而&#xff0c;正确计算和解释当代睡眠EEG中使用的指标需要注意许多理论和实际的信号处理细节。本研究回顾了与频谱分析、蒙太奇选择、相位和…

VC GDI双缓冲绘图

VC GDI双缓冲绘图 VC GDI双缓冲绘图创建内存DC和内存图片&#xff0c;缺一不可最好是封装一下内存绘制绘制效果 关键是不闪烁PS 重绘机制 VC GDI双缓冲绘图 双缓冲绘图&#xff0c;知道这个知识点&#xff0c;每次用的时候还得踩一遍坑&#xff0c;真是服&#xff0c;总结记录…

BGP实验--联邦以及反射器

实验明细 实验拓扑实验要求实验内容 实验拓扑 实验要求 1.R2-R7每台路由器均存在一个环回接口用于建立邻居&#xff1b;同时还存在一个环回来代表连接用户的接口&#xff1b;最终这些连接用户的接口网络需要可以和R1/8的环回通讯 2.AS2网段地址为172.16.0.0/16&#xff0c;减少…

【开源项目】ShenYu网关中Disruptor的使用

模块封装 shenyu-disruptor定义了DisruptorProvider、DisruptorProviderManage、DataEvent、QueueConsumerFactory、DisrutporThreadFactory等一系列通用接口 该模块的搭建了一个disruptor的初始化框架&#xff0c; DisruptorProviderManage提供Disruptor的初始化&#xff0c;…

分布式事务的21种武器 - 4

在分布式系统中&#xff0c;事务的处理分布在不同组件、服务中&#xff0c;因此分布式事务的ACID保障面临着一些特殊难点。本系列文章介绍了21种分布式事务设计模式&#xff0c;并分析其实现原理和优缺点&#xff0c;在面对具体分布式事务问题时&#xff0c;可以选择合适的模式…

软件设计师数据结构速过

加法规则&#xff1a;多项相加&#xff0c;保留最高阶项&#xff0c;并将系数化为 1 乘法规则&#xff1a;多项相乘都保留&#xff0c;并将系数化为 1 加法乘法混合规则&#xff1a;先小括号再乘法规则最后加法规则 时间复杂度估算看最内层循环&#xff0c;如若没有循环和递归则…