MyBatis—操作数据库

news2024/10/7 4:22:17

MyBatis

  • 🔎前置铺垫
    • 创建数据库
    • MyBatis 的执行流程
    • 创建对应流程
  • 🔎MyBatis—查询
    • 查询用户信息
      • 执行流程
      • 创建实体类
      • 创建 Interface 与 xml
      • 在 xml 中编写 SQL 语句
      • 模拟执行流程
  • 🔎单元测试
    • 定义
    • 优点
    • 执行单元测试
      • 引入依赖
      • 生成单元测试
      • 编写代码
      • Assertions—断言
  • 🔎MyBatis—新增
    • 新增用户信息(不返回用户 Id)
      • 在 Interface 中定义方法
      • 在 xml 中实现方法
      • 单元测试验证效果
    • 新增用户信息(返回用户 Id)
      • 在 Interface 中定义方法
      • 在 xml 中实现方法
      • 单元测试验证效果
  • 🔎MyBatis—修改
    • 修改用户信息
      • 在 Interface 中定义方法
      • 在 xml 中实现方法
      • 单元测试验证效果
  • 🔎MyBatis—配置打印执行的 SQL
  • 🔎MyBatis—删除
    • 删除用户信息
      • 在 Interface 中定义方法
      • 在 xml 中实现方法
      • 单元测试验证效果
        • 不污染数据库进行单元测试
  • 🔎对比 # 与 $
    • 根据 id 查询用户信息
    • 根据 username 查询用户信息
    • 根据 id 进行排序
    • 总结
    • SQL 注入
  • 🔎like 查询
    • 根据 username 查询用户信息
    • 使用 # 时的解决办法
  • 🔎类中的属性与数据库的字段名不一致
    • resultMap
    • 重命名
  • 🔎多表查询
  • 🔎动态 SQL 的使用
    • if 标签
    • trim 标签
    • where 标签
    • set 标签
    • foreach 标签
  • 🔎🌸🌸🌸完结撒花🌸🌸🌸

对于 MyBatis 的介绍及创建可以参考这篇文章

🔎前置铺垫


创建数据库


MyBatis 是实现程序和数据库交互的工具

因此在操作 MyBatis 时需要先创建数据库

复制如下代码至 MySQL 中

后续操作将针对该数据库进行演示

-- 创建数据库
drop database if exists excnblog;
create database excnblog DEFAULT CHARACTER SET utf8mb4;

-- 使用数据数据
use excnblog;

-- 创建表[用户表]
drop table if exists  userinfo;
create table userinfo(
    id int primary key auto_increment,
    username varchar(100) not null,
    password varchar(32) not null,
    photo varchar(500) default '',
    createtime timestamp default current_timestamp,
    updatetime timestamp default current_timestamp,
    `state` int default 1
) default charset 'utf8mb4';

-- 创建文章表
drop table if exists  articleinfo;
create table articleinfo(
    id int primary key auto_increment,
    title varchar(100) not null,
    content text not null,
    createtime timestamp default current_timestamp,
    updatetime timestamp default current_timestamp,
    uid int not null, # 发布人
    rcount int not null default 1, # 阅读量
    `state` int default 1
)default charset 'utf8mb4';

-- 创建视频表
drop table if exists videoinfo;
create table videoinfo(
  	vid int primary key,
  	`title` varchar(250),
  	`url` varchar(1000),
	createtime timestamp default current_timestamp,
	updatetime timestamp default current_timestamp,
  	uid int
)default charset 'utf8mb4';

-- 添加用户信息
INSERT INTO `excnblog`.`userinfo` (`id`, `username`, `password`, `photo`, `createtime`, `updatetime`, `state`) VALUES 
(1, 'admin', 'admin', '', '2021-12-06 17:10:48', '2021-12-06 17:10:48', 1);

-- 添加文章信息
insert into articleinfo(title,content,uid)
    values('Java','Java正文',1);
    
-- 添加视频信息
insert into videoinfo(vid,title,url,uid) values(1,'java title','http://www.baidu.com',1);

MyBatis 的执行流程


在这里插入图片描述

创建对应流程


  • 创建 controller 包 → 控制器层
  • 创建 service 包 → 服务层
  • 创建 mapper 包 → 数据持久层
  • 创建 entity 包 → 实体类

在这里插入图片描述

🔎MyBatis—查询


查询用户信息


查询用户信息 → 此处使用的数据表为 userinfo(用户) 表

执行流程


在这里插入图片描述

创建实体类


实体类中的属性对应数据表的字段
(属性名与字段名可以不一一对应, 但建议保持一致)

在这里插入图片描述
在这里插入图片描述

创建 Interface 与 xml


  • Interface → 定义所需的方法
  • xml → 实现定义的方法

创建 Interface🍭

创建 Interface, 需添加@Mapper注解, 目的是为了随着 Spring 项目的启动加载 UserMapper

Interface 中定义的方法的参数建议添加@Param注解

在这里插入图片描述

创建 xml🍭

xml 的配置模板

namespace="", 标识当前的 xml 文件实现的哪个 Interface

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="">

</mapper>

注意对应关系🍂

namespace="com.example.demo.mapper.UserMapper"

表示当前的 xml 文件实现的是 com → example → demo → mapper → UserMapper(Interface)

在这里插入图片描述

在 xml 中编写 SQL 语句


写法1🍂

select * from userinfo where id = ${id}

写法2🍂

select * from userinfo where id = #{id}

大多数情况下使用 #

后面会介绍 $ 与 # 的区别

在这里插入图片描述

注意对应关系🍂

使用<select></select>标签时通常需搭配 2 个属性

  • id, 对应方法名
  • resultType, 对应方法的返回值

在这里插入图片描述

模拟执行流程


服务层(Sevice) → 数据持久层(Mapper)

在这里插入图片描述

控制层(Controller) → 服务层(Service)

设置 URL 时不要大小写组合, 建议全部小写, 利用-_分割

在这里插入图片描述

运行查看结果

在这里插入图片描述

🔎单元测试


定义


单元测试(Unit Testing), 指对软件中的最⼩可测试单元进⾏检查和验证的过程

在 Java 中, 最小可测试单元通常指方法

优点


  • 简单, 直观, 快速的测试某个功能是否正确
  • 可在打包时发现错误(因为打包之前所有单元测试必须通过, 否则无法打包成功)
  • 使用单元测试测试功能时, 可以不污染连接的数据库(即不对数据库进行修改)

执行单元测试


引入依赖


SpringBoot 默认内置了该依赖

<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-test</artifactId>
	<scope>test</scope>
</dependency>

生成单元测试


在 UserMapper(Interface) 中右击 → Generate🍂

在这里插入图片描述

Generate → Test🍂

在这里插入图片描述

点击 Test 弹出如下界面🍂

  • Testing library, 单元测试的框架
  • Class name, 生成的类名
  • Superclass, 测试类继承的父类
  • Destination package, 文件生成的目录名
  • Generate, 生成选项
  • setUp / @Before, 前置选项
  • tearDown / @After, 后置选项
  • Show inherited methods, 显示内置方法

在这里插入图片描述

勾选 getUserById() 方法, 点击 OK

弹出如下界面🍂

在这里插入图片描述

编写代码


  1. 添加注解@SpringBootTest
  2. 在对应的单元测试方法中编写代码

在这里插入图片描述

Assertions—断言


如果断言失败, 后续代码将不会执行

方法说明
Assertions.assertEquals()判断两个对象或两个原始类型是否相等
Assertions.assertNotEquals()判断两个对象或两个原始类型是否不等
Assertions.assertSame()判断两个对象的引用是否指向同一个对象
Assertions.assertNotSame()判断两个对象的引用是否指向不同的对象
Assertions.assertTrue()判断给定的布尔值是否为 true
Assertions.assertFalse()判断给定的布尔值是否为 false
Assertions.assertNull()判断给定的对象引用是否为 null
Assertions.assertNotNull()判断给定的对象引用是否不为 null

🔎MyBatis—新增


使用show create table userinfo查看哪些字段为非空(NOT NULL)

  • username → NOT NULL
  • password → NOT NULL
  • createtime → NOT NULL(但含有默认值)
  • updatetime → NOT NULL(但含有默认值)

在这里插入图片描述

新增用户信息(不返回用户 Id)


新增用户信息 → 此处使用的数据表为 userinfo(用户) 表

新增操作默认返回值为 Int 类型

新增操作的标签为<insert></insert>
使用<insert></insert>标签时需搭配 id 属性

在 Interface 中定义方法


/**
* 新增用户信息
* @author bibubibu
* @date 2023/7/2
*/
int add(Userinfo userinfo);

在这里插入图片描述

在 xml 中实现方法


<insert id="add">
    insert into userinfo(username, password) values(#{username}, #{password})
</insert>

在这里插入图片描述

注意🍂

对于values(#{username}, #{password})

username 对应的是实体类中的属性(不是字段名)
password 对应的是实体类中的属性(不是字段名)

在这里插入图片描述

单元测试验证效果


在这里插入图片描述

非第一次添加单元测试时, 可能会出现 Error

点击 OK 即可

在这里插入图片描述


测试查看效果

在这里插入图片描述
在这里插入图片描述

针对当前代码, 无法得到添加对象成功后对象的 id🍂

userinfo 表中的 id 是自增的

对于当前运行结果, 表示已添加成功, 但无法得到添加对象的 id

在这里插入图片描述

新增用户信息(返回用户 Id)


返回用户 Id, 需在<insert></insert>标签添加useGeneratedKeyskeyProperty

  • useGeneratedKeys, 是否使用生成的主键
  • keyProperty, 将生成的主键赋值给哪个属性

在 Interface 中定义方法


/**
* 新增用户信息(返回用户 Id)
* @author bibubibu
* @date 2023/7/2
*/
int addAndRetId(UserInfo userInfo);

在这里插入图片描述

在 xml 中实现方法


<insert id="addAndRetId" useGeneratedKeys="true" keyProperty="id">
    insert into userinfo(username, password) values(#{username}, #{password})
</insert>

注意🍂

keyProperty="id"对应的是类中的属性, 而非数据库中的字段

  • Property, 属性
  • Column, 字段

在这里插入图片描述

单元测试验证效果


在这里插入图片描述

在这里插入图片描述

🔎MyBatis—修改


修改用户信息


修改用户信息 → 此处使用的数据表为 userinfo(用户) 表

修改操作默认返回值为 Int 类型

修改操作的标签为<update></update>
使用<update></update>标签时需搭配 id 属性

在 Interface 中定义方法


/**
* 修改用户信息
* @author bibubibu
* @date 2023/7/3
*/
int updateUserInfo(UserInfo userInfo);

在这里插入图片描述

在 xml 中实现方法


<update id="updateUserInfo">
    update userinfo set username = #{username}, password = #{password} where id = #{id}
</update>

在这里插入图片描述

单元测试验证效果


在这里插入图片描述

在这里插入图片描述

🔎MyBatis—配置打印执行的 SQL


# 设置 Mybatis 的 xml 保存路径
mybatis:
  mapper-locations: classpath:mapper/*Mapper.xml
  configuration: # 配置打印 MyBatis 执行的 SQL
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
    
# 设置日志级别
logging:
  level:
    com:
      example:
        demo: debug

在这里插入图片描述

效果测试🍭

在这里插入图片描述

🔎MyBatis—删除


删除用户信息


删除用户信息 → 此处使用的数据表为 userinfo(用户) 表

删除操作默认返回值为 Int 类型

删除操作的标签为<delete></delete>
使用<delete></delete>标签时需搭配 id 属性

在 Interface 中定义方法


/**
* 删除用户信息
* @author bibubibu
* @date 2023/7/3
*/
int delUserInfo(@Param("id") Integer id);

在这里插入图片描述

在 xml 中实现方法


<delete id="delUserInfo">
    delete from userinfo where id = #{id};
</delete>

在这里插入图片描述

单元测试验证效果


在这里插入图片描述

在这里插入图片描述

不污染数据库进行单元测试


利用注解@Transactional

注解@Transactional既可以修饰类, 也可以修饰方法

在这里插入图片描述

举个栗子🌰

要求

测试删除功能是否正常 + 数据库中的数据不会被删除
(即不污染数据库进行单元测试)

此时可以利用注解@Transactional

删除功能正常🍂

在这里插入图片描述

数据库中的数据不会被删除🍂

在这里插入图片描述

🔎对比 # 与 $


根据 id 查询用户信息


/**
* 根据 Id 查询用户信息
* @author bibubibu
* @date 2023/7/2
*/
UserInfo getUserById(@Param("id") Integer id);

使用 #🍂

<select id="getUserById" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo where id = #{id}
</select>

在这里插入图片描述

使用 $🍂

<select id="getUserById" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo where id = ${id}
</select>

在这里插入图片描述

运行结果🍂

均可正常运行

根据 username 查询用户信息


/**
* 根据 username 查询用户信息
* @author bibubibu
* @date 2023/7/3
*/
List<UserInfo> getUserByName(@Param("username") String username);

使用 #🍂

<select id="getUserByName" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo where username = #{username}
</select>

在这里插入图片描述

使用 $🍂

<select id="getUserByName" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo where username = ${username}
</select>

在这里插入图片描述

运行结果🍂

使用 # 正常运行
使用 $ 无法正常运行

根据 id 进行排序


排序有 2 种类型

  • 升序 → asc
  • 降序 → desc

此处以降序举例

/**
* 根据 Id 进行排序
* @author bibubibu
* @date 2023/7/3
*/
List<UserInfo> getUserOrderById(@Param("order") String order);

使用 #🍂

<select id="getUserOrderById" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo order by id #{order}
</select>

在这里插入图片描述

使用 $🍂

<select id="getUserOrderById" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo order by id ${order}
</select>

在这里插入图片描述

运行结果🍂

使用 # 无法正常运行
使用 $ 正常运行

总结


  • #{} → 编译预处理, 不存在 SQL 注入问题
  • ${} → 直接替换, 存在 SQL 注入问题

使用 $ 的注意事项🍂

一定是可穷举的值(例如关键字…), 在使用之前要对传递的值进行安全性验证


编译预处理是指 MyBatis 在处理 #{ } 时, 将 SQL 中的 #{ } 替换为 ?

直接替换是指 MyBatis 在处理 #{ } 时, 将 SQL 中的 #{ } 替换为参数的值

  • 根据 id 查询用户信息🍂
    • #{} → 将#{id}替换为?, 参数为 Integer id, 赋值 ? 为对应类型的值(数值型数据无需'')
    • ${} → 将${id}替换为对应的 Integer id 的值
      • 即针对数值形数据, #{}无需添加''(也可理解为直接替换的一种方式), 因此利用#可以运行成功
      • select * from userinfo where id = 1(✔)
      • 即针对数值形数据, ${}直接替换, 因此利用$可以运行成功
      • select * from userinfo where id = 1(✔)
  • 根据 username 查询用户信息🍂
    • #{} → 将#{username}替换为?, 参数为 String username, 赋值 ? 为对应类型的值(字符串型数据需'')
    • ${} → 将${username}替换为对应的 String username 的值
      • 即针对字符串型数据, #{}会添加'', 因此利用#可以运行成功
      • select * from username where username = 'Tom'(✔)
      • 即针对字符串型数据, ${}直接替换, 因此利用$无法运行成功
      • select * from username where username = Tom(✘)
  • 根据 id 进行排序🍂
    • #{} → 将#{order}替换为?, 参数为 String order, 赋值 ? 为对应类型的值(字符串型数据需'')
    • ${} → 将${order}替换为对应的 String order 的值
      • 即针对字符串型数据, #{}会添加'', 因此利用#无法运行成功(关键字无需添加'')
      • select * from userinfo order by id 'desc'(✘)
      • 即针对字符串型数据, ${}直接替换, 因此利用$可以运行成功(关键字无需添加'')
      • select * from userinfo order by id desc(✔)

简单来说就是针对不同的内容

#{}可能会对参数添加''(字符串型), 也可能是直接替换(数值型)

${}直接替换为对应参数的内容


SQL 注入


既然 # 对于字符串型数据会添加'', 那使用 $ 时对于字符串型手动添加''不可以么


使用 $ 是不安全的, 可能会引起 SQL 注入问题


什么是 SQL 注入?

举个栗子🌰

userinfo 表中的内容🍂

在这里插入图片描述

定义一个 login 方法🍂

UserInfo login(@Param("username") String username, @Param("password") String password);

实现该方法🍂

(对于字符串型手动添加'')

<select id="login" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo where username = '${username}' and password = '${password}'
</select>

单元测试查看结果🍂

登录成功

在这里插入图片描述

修改密码后查看结果🍂

登录失败

在这里插入图片描述

利用 SQL 注入登录(使用错误的密码)🍂

登录成功

在这里插入图片描述

注意此时的密码

String password = "' or 1 = '1";

最终生成的 SQL 语句

select * from userinfo where username = 'admin' and password = '' or 1 = '1'

划分为 2 部分(优先级 → and > or)

username = 'admin' and password = '' / 1 = '1'

因此可以成功登录

🌸上述即为 SQL 注入, 因此推荐使用 # 而不是 $🌸


🔎like 查询


like 查询 → 此处使用的数据表为 userinfo(用户) 表

数据表中的数据

在这里插入图片描述

根据 username 查询用户信息


在 Interface 中定义方法🍂

/**
* Like 查询
* @author bibubibu
* @date 2023/7/3
*/
List<UserInfo> getListByName(@Param("username") String username);

在这里插入图片描述

在 xml 中实现方法🍂

<select id="getListByName" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo where username like '%#{username}%'
</select>

在这里插入图片描述

单元测试验证效果🍂

在这里插入图片描述

Error → 这是因为使用 # 处理字符串类型时, SQL 语句会添加''

即上述 SQL 语句变为select * from userinfo where username like '%'a'%'

期望的效果是select * from userinfo where username like '%a%'

可能有的小伙伴会说, 那可以用 $ 处理呀


不可以, 还记得上面说的使用 $ 时的注意事项么
使用 $ 时的注意事项 → 一定是可穷举的值(例如关键字…), 而 username 无法穷举, 因此应避免使用 $

使用 # 时的解决办法


解决办法 → 利用 concat() 进行拼接

<select id="getListByName" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo where username like concat('%', #{username}, '%')
</select>

对比🍂

select * from userinfo where username like '%#{username}%'

select * from userinfo where username like concat('%', #{username}, '%')

单元测试验证效果🍂

在这里插入图片描述

🔎类中的属性与数据库的字段名不一致


在这里插入图片描述

数据表中的数据

在这里插入图片描述

查询数据

/**
* 查询全部信息
* @author bibubibu
* @date 2023/7/2
*/
List<UserInfo> getAll();
<select id="getAll" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo
</select>

在这里插入图片描述

分析结果发现类中的属性名与数据库的字段名不一致的均未被赋值

解决方法🍭

  1. resultMap
  2. 重命名

使用较多的是重命名

resultMap


使用resultMap将数据库中的字段与类中的属性映射

在这里插入图片描述

使用resultMap🍂

<select id="getAll" resultMap="bibubibu">
    select * from userinfo
</select>

问题🍂

并非所有的属性与字段名称都不一致, 需要全部进行映射么?

是的, 仅修改不一致的字段在单表查询时可能不会问题, 但多表查询时可能会出现 null


测试结果🍂

在这里插入图片描述

重命名


重命名, 更为简便的映射属性与字段

<select id="getAll" resultType="com.example.demo.entity.UserInfo">
    select id, username as name, password as pwd, photo, createtime, updatetime, state from userinfo;
</select>

测试结果🍂

在这里插入图片描述

🔎多表查询


多表查询的实现 → 联表查询(left join / right join / inner join) + xxxVO

举个栗子🌰

查询文章的作者(username)

此处查询的数据表为 userinfo(用户) 表,articleinfo(文章) 表

由于 articleinfo(文章) 表只有 uid(用户 id), 因此需要进行多表联查

下面为具体步骤


articleinfo 表与 userinfo 表🍂

articleinfo 表中的 uid 对应 userinfo 表中的 id

在这里插入图片描述

根据 articleinfo 表创建 ArticleInfo 实体类🍂

在这里插入图片描述

在这里插入图片描述

多表联查时返回数据需要包含文章作者(username)
因此扩展 articleinfo(文章) 表的实体类

扩展的实体类放入 objectview 包下, 继承 Articleinfo 类

在这里插入图片描述

在 Interface 中定义方法🍂

ArticleInfoVO getById(@Param("id") Integer id);

在这里插入图片描述

在 xml 中实现方法🍂

<select id="getById" resultType="com.example.demo.entity.viewobject.ArticleInfoVO">
    select a.*, u.username from articleinfo as a
    left join userinfo as u
    on a.uid = u.id
    where a.id = #{id}
</select>

在这里插入图片描述

单元测试验证效果🍂

发现只有 username 字段, 其他字段消失了

在这里插入图片描述

Debug 分析错误🍂

发现查询的结果是包含 username + 其他字段的

在这里插入图片描述

排查错误🍂

发现是 Lombok 重写 toString() 方法时的 BUG

在这里插入图片描述

重写 toString()🍂

在这里插入图片描述

在这里插入图片描述

单元测试验证效果🍂

在这里插入图片描述

另一种解决方法🍂

取消继承, 重写全部属性

在这里插入图片描述
在这里插入图片描述

注意事项🍂

实体类建议实现 Serializable 接口

目的是为了完成该类的序列化与反序列化操作

在这里插入图片描述

🔎动态 SQL 的使用


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

官方对于动态 SQL 的介绍

if 标签


在数据表中
一些字段必须填写(必填字段)
一些字段可以填写, 也可以不填写(非必填字段)

如果在编写 SQL 语句中有不确定的字段(非必填字段), 就需要使用 if 标签进行判断

语法🍂

<if test="">
            
</if>

举个栗子🌰

SQL 语句

insert into userinfo(username, password, photo) values('Lily', '6666', 'Lily.png');

xml 中实现

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

trim 标签


如果所有字段都是非必填字段, 通常使用 trim 标签 + if 标签

trim 标签的属性🍂

  • prefix, 表示整个代码块以 prefix 的值作为前缀
  • suffix, 表示整个代码块以 suffix 的值作为后缀
  • prefixOverrides, 表示整个代码块要去除的前缀
  • suffixOverrides, 表示整个代码块要去除的后缀

语法🍂

<trim prefix="" suffix="" prefixOverrides="" suffixOverrides="">
            
</trim>

举个栗子🌰

SQL 语句

insert into userinfo(username, password, photo) values('Judy', 'Judy_pwd', 'Judy.png');

xml 中实现

<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>

where 标签


where 标签通常搭配 if 标签一起使用

where 标签的特点🍂

  • where 标签会删除最前面的 and 关键字
  • 如果 where 标签中没有内容(即 if 属性中的参数全部为 null), 就不会生成相对应的 where 语句

语法🍂

<where>
        
</where>

举个栗子🌰

SQL 语句

select * from userinfo where username = 'Tom' and password = 'Tom_pwd';

xml 中实现

<select id="getByCondition" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo
	<where>
        <if test="username != null">
            username = #{username}
        </if>
        <if test="password != null">
            and password = #{password}
        </if>
    </where>
</select>

利用 trim 标签实现 where 标签🍂

<select id="getByCondition" resultType="com.example.demo.entity.UserInfo">
    select * from userinfo
	<trim prefix="where" prefixOverrides="and">
         <if test="username != null">
             username = #{username}
         </if>
         <if test="password != null">
             and password = #{password}
         </if>
     </trim>
</select>

set 标签


set 标签通常搭配 if 标签一起使用

set 标签的特点🍂

  • set 标签会删除最后面的,

语法🍂

<set>
        
</set>

举个栗子🌰

SQL 语句

update userinfo set username = 'Jackson', password = 'Jackson_pwd', photo = 'Jackson.png' where id = 9;

xml 中实现

<update id="update">
    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>

利用 trim 标签实现 where 标签🍂

<update id="update">
    update userinfo
    <trim prefix="set" suffixOverrides=",">
         <if test="username != null">
             username = #{username},
         </if>
         <if test="password != null">
             password = #{password},
         </if>
         <if test="photo != null">
             photo = #{photo},
         </if>
     </trim>
    where id = #{id}
</update>

foreach 标签


foreach 标签通常用于遍历集合

foreach 标签常用属性🍂

  • collection, 绑定方法中的参数, 如 List, Set, Map 或数组对象
  • item, 遍历时的每一个对象
  • open, 代码块开头的字符串
  • close, 代码块结束的字符串
  • separator, 每次遍历之间间隔的字符串

语法🍂

<foreach collection="" open="" close="" item="" separator="">

</foreach>

举个栗子🌰

SQL 语句

delete from userinfo where id in(1, 2, 3);

xml 中实现

<delete id="dels">
    delete from userinfo where id in
    <foreach collection="ids" open="(" close=")" item="val" separator=",">
        #{val}
    </foreach>
</delete>

对应关系🍂

在这里插入图片描述

🔎🌸🌸🌸完结撒花🌸🌸🌸


在这里插入图片描述

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

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

相关文章

优思学院|制造不良品是品质成本意识的问题吗?

制造的最大错误&#xff0c;就是制造出不良品或者不适合的物品。 谁都知道制造出这类“不良品”会使成本增加&#xff0c;但是到底会增加什么成本&#xff0c;可能就不是人人 都清楚。 若是不小心制造出不适合品&#xff0c;浪费的成本除了该物品的材料费、加工人力的费用、设…

VR电力安全警示教育:身临其境体验事故伤害

VR电力安全警示教育由广州华锐互动开发&#xff0c;是一种利用虚拟现实技术模拟电厂安全事故的应用程序。该应用程序通过模拟真实的场景和情境&#xff0c;让用户身临其境地体验VR电力安全警示教育的过程&#xff0c;从而提高用户的安全意识和应对能力。 在VR电力安全警示教育中…

Linux下获取另外一个程序的标准输出和标准错误输出的一种实现方式

问题&#xff1a;一个程序如何获取另外一个程序的标准输出和标准错误输出&#xff1f; 标准输入&#xff0c;标准输出&#xff0c;标准错误输出是一个程序的基本组成&#xff0c;在Linux下一个程序调用另外一个程序&#xff0c;如何获取其标准输出和错误输出呢&#xff1f; 分析…

专用开发工具和环境在嵌入式系统开发中的重要性

嵌入式系统是一种特殊类型的计算机系统&#xff0c;其设计和开发用于控制特定设备或执行特定任务。以下是嵌入式系统的一些特点以及与其开发相关的专用工具和环境&#xff1a; 特定目标&#xff1a;嵌入式系统通常被设计用于执行特定的任务或控制特定的设备。它们被定制以满足特…

开发一款财经直播系统需要注意的方面

数据保护与隐私&#xff1a;确保用户的个人信息和交易数据得到有效的保护&#xff0c;遵守相关的数据保护法规。采用高级加密技术来保护用户的敏感信息&#xff0c;并建立安全的存储和传输机制。 信息真实性&#xff1a;提供准确、可信的财经信息&#xff0c;确保平台上发布…

【Jenkins】Jenkins构建后端流水线

目录 一、新建任务1、输入任务名称&#xff0c;选择构建项目类型&#xff08;这里我选择的是Maven项目&#xff09;&#xff0c;任务名称一般格式为&#xff1a;项目名称-前后端2、创建成功后的结果 二、配置流水线1、进入刚创建好的任务页面中&#xff0c;点击配置2、General配…

Loki+Grafana监控docker容器日志

目标&#xff1a;最近开发人员时常需要查看各个环境项目中容器日志&#xff0c;而直接通过ssh终端使用docker logs命令查看日志不太安全&#xff0c;这会导致很多人员知道服务器的账户和密码&#xff0c;有没有一种可以收集所有容器日志的平台系统。那么这套系统就是LokiGrafan…

3.3C++输入流

C 输入概述 C输入是指程序从外部获取数据的过程。 C提供多种输入方法&#xff0c;包括从键盘输入、从文件读取、从网络获取&#xff0c;最常用的是从键盘输入。 C输入数据的函数是cin&#xff0c;它可以读取各种类型的数据&#xff0c;包括整数、浮点数、字符和字符串等。 …

7.1.8 其他Linux支持的文件系统与VFS

虽然 Linux 的标准文件系统是 ext2 &#xff0c;且还有增加了日志功能的 ext3/ext4 &#xff0c;事实上&#xff0c;Linux 还有支持很多文件系统格式的&#xff0c; 尤其是最近这几年推出了好几种速度很快的日志式文件系统&#xff0c;包括 SGI 的 XFS 文件系统&#xff0c; 可…

css文字超出元素省略,单行、多行省略

效果图: 通用CSS .box {width: 500px;border: 1px solid red;padding: 10px;line-height: 24px;}1.单行省略 .singe-line {text-overflow: ellipsis;overflow: hidden;word-break: break-all;white-space: nowrap;}<p>单行省略</p><div class"singe-lin…

如何统一接口测试的功能、自动化和性能测试用例

服务端的测试&#xff0c;大多数内容都围绕着接口展开。对于接口测试&#xff0c;无非也是功能、自动化、性能测试为主&#xff0c;偶然想到一个问题&#xff0c;如果能写一个用例&#xff0c;在功能、自动化、性能三者的测试场景中重复使用&#xff0c;肯定能省去非常多的时间…

webpack相关面试题

webpack面试题 1.webpack和vite区别2.如何优化webpack打包速度&#xff1f;3.说说webpack中常见的Plugin&#xff1f;解决了什么问题4.说说如何借助webpack来优化前端性能&#xff1f;如何优化JS代码压缩CSS代码压缩Html文件代码压缩文件大小压缩图片压缩Tree ShakingusedExpor…

中小企业的数字化热情,从未像今年618这样滚烫

一年一度的618大促告一段落&#xff0c;我们可以总结一个趋势&#xff1a;C端消费者对大促的热情在消退&#xff0c;而B端企业&#xff0c;尤其是中小企业&#xff0c;对大促的热情则以肉眼可见的速度提升。 普通消费者也好&#xff0c;广大中小企业也罢&#xff0c;参与大促的…

SSMP整合案例(11) 在界面中实现添加操作

上文 SSMP整合案例(10) vue端调整项目环境 发送请求 基本界面编写我们搭建了基本的页面结构 然后 我们来做个新增的功能 首先 新增 我们肯定是用户点击了这个新建之后 我们再来处理这个逻辑 我们之前的代码 新增是有绑定 一个事件的 但是这个 AddBook中并没有内容 首先 我们…

想要经营好抖音小店,你还需要了解这些活动方法

抖音小店作为抖音上的新零售业务形态&#xff0c;旨在为用户提供更加便捷的购物体验&#xff0c;同时也为商家提供了一个新的销售渠道。在使用抖音小店的过程中&#xff0c;有一些主流的玩法&#xff0c;今天不若与众科技就来介绍一下。 1. 抖音小店开通直播带货服务 开通直播…

PowerToys工具介绍及使用

目录 介绍下载使用和功能介绍1.安装后的效果2.始终置顶3.唤醒4.颜色选择器5.FancyZones 窗口分屏器6.File Locksmith7.文件资源管理器加载项8.Host文件编辑器9.图像大小调整器10.键盘管理器11.速览12.PowerRename13.文本提取器 介绍 PowerToys 是微软最初发布于 Windows 95 平…

计算机毕业论文选题推荐|软件工程|小程序系列选题

文章目录 导文题目导文 计算机毕业论文选题推荐|软件工程 (***语言)==使用其他任何编程语言 例如:基于(***语言)门窗账务管理系统的设计与实现 得到:基于JAVA门窗账务管理系统的设计与实现 基于vue门窗账务管理系统的设计与实现 等等 题目 基于微信小程序和深度学习的宠物…

Minio设置文件链接永久有效的完整步骤

目录 前言1.下载MinIO Client2.运行MinIO Client3.添加一个云存储服务4.验证5.policy命令 - 管理存储桶策略总结 前言 minio分享文件的链接&#xff0c;最多支持分享七天 通过 MinIO客户端 管理存储桶策略的方式实现文件链接永久有效 1.下载MinIO Client 采用 Docker 方式 …

拓展销售渠道是客户管理系统的重要功能

随着市场竞争的加剧&#xff0c;人们对于未来的期望和追求越来越高。科技的飞速发展和全球化的趋势&#xff0c;使得我们的生活方式和工作方式都发生了翻天覆地的变化。企业需要不断拓展销售渠道&#xff0c;以吸引更多的客户和提高销售额。客户管理系统&#xff08;CRM&#x…

性能测试岗位常见面试题及答案,希望可以帮到你

目录 前言 一、基础篇 二、工具篇 三、系统架构篇 四、服务器&中间件篇 五、数据库篇 七、案例篇 总结&#xff1a; 前言 最近有童鞋找我问一些性能测试相关的问题&#xff0c;其中问的次数最多的&#xff0c;还是面试性能测试岗位&#xff0c;一般会问哪些问题&am…