MyBatis - 单元测试 参数传递 注解 CRUD

目录

1. MyBatis 简介

2. 简单使用 MyBatis

2.1 创建 MyBatis 项目

2.2 连接数据库

2.3 创建 Java 类

2.4 创建 Mapper 接口

 2.5 在测试类中执行

3. 单元测试

3.1 @Test

3.2 @SpringBootTest

3.3 @BeforeEach / @AfterEach

4. MyBatis 基础操作

4.1 配置 MyBatis 打印日志

4.2 参数传递

4.2.1 传递单个参数

4.2.2 传递多个参数

4.2.2.1 @Param: 参数绑定/重命名

4.2.2 接收结果

5. 新增数据(@Insert)

5.1 @Insert

5.2 @Options: 返回主键

6. 查询数据(@Select)

6.1 解决映射问题

6.1.1 表中字段起别名

​编辑6.1.2 @Results 结果映射

6.1.3 配置驼峰自动转换 (推荐)

7. 更新数据(@Update)

8. 删除数据(@Delete)


1. MyBatis 简介

我们之前提到过应用分层:

  1. Controller => 控制/表现层, 用于和用户交互(接收参数, 返回结果)
  2. Service => 业务逻辑层, 处理业务数据
  3. Dao => 数据层, 对数据进行管理和查询

MyBatis 就是一款优秀的 Dao 层框架.

我们之前学习过 jdbc 操作数据库, 操作繁琐麻烦, 并且每次操作都需要写很多重复的代码, 如: 创建 DataSource, 获取连接, 释放资源, ....

MyBatis 就是基于 jdbc 封装而成的框架, 就类似于 Spring 是基于 serverLet 封装的一样.

也就是说, MyBatis 是一款操作数据库的框架, 它极大的简化了操作数据库的流程, 能够让我们写更少的代码, 更容易的操作数据库.

注意: MyBatis 底层仍然使用 jdbc 来操作数据库, 但我们不再使用 jdbc, 只需操作 MyBatis, 让 MyBatis 操作 jdbc 即可.

2. 简单使用 MyBatis

首先, 我们先使用 sql 创建 mybatis_test 数据库, 再创建 user_info 表, 在表中插入数据.

数据插入成功后, 如下图所示:

接着, 我们就可以使用 MyBatis 操作 mybatis_test 数据库了.

-- 创建数据库
DROP DATABASE IF EXISTS mybatis_test;CREATE DATABASE mybatis_test DEFAULT CHARACTER SET utf8mb4;-- 使用数据数据
USE mybatis_test;-- 创建表[用户表]
DROP TABLE IF EXISTS user_info;
CREATE TABLE `user_info` (`id` INT ( 11 ) NOT NULL AUTO_INCREMENT,`username` VARCHAR ( 127 ) NOT NULL,`password` VARCHAR ( 127 ) NOT NULL,`age` TINYINT ( 4 ) NOT NULL,`gender` TINYINT ( 4 ) DEFAULT '0' COMMENT '1-男 2-女 0-默认',`phone` VARCHAR ( 15 ) DEFAULT NULL,`delete_flag` TINYINT ( 4 ) DEFAULT 0 COMMENT '0-正常, 1-删除',`create_time` DATETIME DEFAULT now(),`update_time` DATETIME DEFAULT now() ON UPDATE now(),PRIMARY KEY ( `id` ) 
) ENGINE = INNODB DEFAULT CHARSET = utf8mb4; -- 添加用户信息
INSERT INTO mybatis_test.user_info( username, `password`, age, gender, phone )
VALUES ( 'admin', 'admin', 18, 1, '18612340001' );
INSERT INTO mybatis_test.user_info( username, `password`, age, gender, phone )
VALUES ( 'zhangsan', 'zhangsan', 18, 1, '18612340002' );
INSERT INTO mybatis_test.user_info( username, `password`, age, gender, phone )
VALUES ( 'lisi', 'lisi', 18, 1, '18612340003' );
INSERT INTO mybatis_test.user_info( username, `password`, age, gender, phone )
VALUES ( 'wangwu', 'wangwu', 18, 1, '18612340004' );-- 创建文章表
DROP TABLE IF EXISTS article_info;CREATE TABLE article_info (id INT PRIMARY KEY auto_increment,title VARCHAR ( 100 ) NOT NULL,content TEXT NOT NULL,uid INT NOT NULL,delete_flag TINYINT ( 4 ) DEFAULT 0 COMMENT '0-正常, 1-删除',create_time DATETIME DEFAULT now(),update_time DATETIME DEFAULT now() 
) DEFAULT charset 'utf8mb4';-- 插入测试数据
INSERT INTO article_info ( title, content, uid ) VALUES ( 'Java', 'Java正文', 1 );

2.1 创建 MyBatis 项目

使用 MyBatis, 依旧创建 SpringBoot 项目即可.

要进行说明的是, Spring 和 MyBatis 是 "两家人", 是两个不同的框架, 但是由于 MyBatis 功能强大, 于是 Spring 集成了 MyBatis, 就像 12306 集成了美团的外卖服务和其他公司的保险服务一样.

接下来, 我们创建 SpringBoot 项目, 并通过插件导入 MyBatis 依赖:

2.2 连接数据库

首先, 需要进行数据库连接的配置(yml 格式为例):

# 数据库配置
spring:datasource:url: jdbc:mysql://127.0.0.1:3306/mybatis_test?characterEncoding=utf8&useSSL=falseusername: rootpassword: 111111driver-class-name: com.mysql.cj.jdbc.Driver

以上配置数据的主要功能为: 将 Java 进程连接到本地运行的名为 mybatis_test 的 MySQL 数据库, 用户名为 root, 密码为 111111.
(注意: 避免出错, 不要手写!! 直接复制到配置文件中)

2.3 创建 Java 类

这里, 我们使用 MyBatis 执行最简单的 select 语句.

MyBatis 会将查询结果绑定到 Java 对象的属性中, 这里我们使用 UserInfo 对象接收.

import lombok.Data;
import java.util.Date;@Data
public class UserInfo {private Integer id;private String username;private String password;private Integer age;private Integer gender;private String phone;private Integer deleteFlag;private Date createTime;private Date updateTime;
}

2.4 创建 Mapper 接口

对接口使用 @Mapper 注解, 告诉 MyBatis 这是一个 Mapper 接口, 并让 Spring 管理该接口的实现类.

在 Mapper 接口中, 会定义数据库操作的方法, 例如: 在 selectAll 方法的 @Select 注解中指定要执行的 SQL.

综上:

  • @Mapper: 将接口的实现类的对象交给 Spring 管理
  • @Select: 定义 SQL 语句, 当调用该方法时会执行其中的 SQL 语句
@Mapper
public interface UserInfoMapper {@Select("select * from user_info")List<UserInfo> selectAll();
}

 2.5 在测试类中执行

最后, 在测试类中进行测试, 观察 MyBatis 执行结果.

@SpringBootTest: 为该测试类提供 Spring 运行环境

@Autowired 获取实现类 Bean 后, 实现类 Bean 调用 SelectAll 方法, MyBatis 执行该方法 @Select 注解中的 SQL, 并将查询得到的结果集的每一行映射到每一个 UserInfo 对象的属性中, 并将每一个 UserInfo 对象放入 List 中.

3. 单元测试

在之前的学习中, 我们想要测试所写代码是否正确, 我们通常都会在 main 方法中通过打印来进行测试. 其实, 正确的做法应该是在测试类中进行测试, 这样可以将项目代码和测试代码分离开来, 一目了然的区分项目代码和测试代码.

这里所说的测试代码不是测试人员进行测试所写的代码, 而是我们开发人员进行单元测试时写的带代码, 毕竟项目的第一个测试人员, 一定是开发者自己.

其实, 在上文简单使用 MyBatis 时, 就已经用到了测试类, 接下来, 为大家详细讲一讲.

3.1 @Test

在测试类中, 使用 @Test 注解标记方法, 该方法可以直接运行, 不需要通过调用就可以直接执行:

3.2 @SpringBootTest

@SpringBootTest 的作用是: 为测试类加载 Spring 运行环境, 它会启动一个完整的 Spring 应用上下文, 也就是说我们可以在测试类中使用 Spring 的所有特性.

当我们需要在测试类中使用到 Spring 的相关功能时(如: @Autowired 依赖注入), 就需要给类加上 @SpringBootTest. (如果不需要使用 Spring 的功能, 就可以不加, 如上文 @Test 所示举例)

3.3 @BeforeEach / @AfterEach

@BeforeEach: 在每一个测试方法执行前, 执行的方法

@AfterEach: 在每一个测试方法执行后, 执行的方法

 

4. MyBatis 基础操作

4.1 配置 MyBatis 打印日志

配置 MyBatis 打印日志后, 我们可以借助日志查看注解的 SQL 语句, 参数的传递以及 SQL 的执行结果. 

# 配置打印 MyBatis⽇志
mybatis:configuration: log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

4.2 参数传递

上文, 我们演示了如何使用 @Select 查询标准所有的记录, 但是在绝大多数的查询时, 我们都是会加上 where 来锁定我们要查询的目标的.

此时, 我们就需要将参数传递到 SQL 语句中, 以到达我们查询的目的.

4.2.1 传递单个参数

在 MyBatis 中, 使用 #{参数名称} 的形式来进行参数的传递.

注意: 此时只传递了一个参数, 因此 #{} 中的名称可以任意命名, 无论是什么名称都会绑定成功:

4.2.2 传递多个参数

同样使用 #{参数名称} 的形式来传递参数:

注意: 此时, 传递的参数有多个, MyBatis 会根据 #{} 中参数的名称来匹配方法中的形参.

和参数顺序无关, 即使调换形参的顺序, MyBatis 也会根据参数名称进行匹配:

但是, 由于此时传递的是多个参数, 因此 #{} 中的参数不可随意命名, 名称必须和方法形参的名称相匹配, 否则匹配失败报错:

如上图, #{} 中除了使用形参名称和形参进行匹配外, 还可以根据形参顺序使用 param1, param2, ... 进行匹配, 但不建议, 因为可读性差.

4.2.2.1 @Param: 参数绑定/重命名

上文说到, 传递多个参数时, #{} 中的参数名称必须和方法形参的名称相匹配. 

#{} 中的名称除了和形参名称相同能够匹配成功外, 还可以使用 @Param 自定义参数名(对方法形参使用), 进行参数绑定.

综上, 方法形参和参数占位符(#{})能够匹配成功的方式有两种:

  1. 两者参数使用相同的名称
  2. 使用 @Param 进行参数绑定

4.2.2 接收结果

这里以 select 操作为例:

  1. 当查询返回的结果集只有 0 条或 1 条记录时, 使用对象或者集合接收均可.
  2. 当查询返回的结果集有多条记录时, 必须使用集合接收.

5. 新增数据(@Insert)

5.1 @Insert

使用 MyBatis 进行数据插入时, 需要使用 @Insert 注解.

由于插入数据时, 需要给表中的多个字段赋值, 为了简化代码, 我们可以直接给方法传一个对象(传单个参数也是可以的), MyBatis 会根据对象中属性的名称, 自动匹配到对应的参数占位符(#{})上.

此外, 上一篇博客说到, 可以使用 @Param 进行参数绑定, 那么我们对对象使用 @Param 观察结果: 

我们发现, 对象使用 @Param 进行参数绑定后, 程序就会出错.

我们需要这样进行修改:

5.2 @Options: 返回主键

通过 @Insert 插入数据后, 还可以通过 @Options 获取新增数据的主键值(ID).

6. 查询数据(@Select)

在上文中, 已经讲解了 MyBatis 进行数据查询的大部分内容, 但是遗留了一个问题:

当我们打印 @Select 的查询结果时, 有些属性的值为 null:

原因其实很简单: 由于数据库表中字段的名称和 Java 类中属性的名称不相同, 导致值没有映射成功:

6.1 解决映射问题

解决办法有三种:

  1. 查询时, 给表中字段起别名, 使表中字段和 Java 属性名称相同
  2. 使用 @Results
  3. 修改配置, 使小驼峰和蛇形名称相绑定

6.1.1 表中字段起别名

查询时, 给表中字段起别名, 使得查询结果集中的字段名称和 Java 属性名称相同:

这样, 就能够映射成功了:

6.1.2 @Results 结果映射

虽然给字段起别名的方式能够映射成功, 但是这种方式太麻烦了. 有一种比较简洁的方式: 使用 @Results 注解.

@Results 中的 value 属性是 @Result[] 类型的 , @Result 便可以设置数据库字段和 Java 属性之间的映射关系. 由于 value 是 @Result[] 类型即数组的类型的, 因此可以设置多组的映射关系:

6.1.3 配置驼峰自动转换 (推荐)

虽然 @Results 的方式已经够简单了, 但是, 还有更加 "懒汉" 的方式 --- 

其实, Java 属性和 数据库字段映射不上的根本原因就是: 两者命名规范不同.

Java 属性使用小驼峰命名, 数据库字段使用蛇形命名(_), 但两者参数名称都是很相似的, 因此, 我们只需在配置文件中, 将小驼峰的格式和蛇形格式绑定到一起即可:

# 配置驼峰⾃动转换
mybatis:configuration:map-underscore-to-camel-case: true 

这样, 我们不需使用任何注解, 就能够将数据库中的蛇形字段和 Java 中的小驼峰属性绑定到一起了:

因此, 在对 Java 属性进行命名时, 如果可以和数据库字段名称完全一致则完全一致, 如果由于命名规范的限制而无法完全一致, 则 Java 属性名称应使用数据库字段名称相同单词的小驼峰形式, 以便通过配置文件能够成功映射.

7. 更新数据(@Update)

使用 MyBatis 进行 update 操作, 需要使用 @Update 注解. 

同样, 进行数据更新时, 既可以传入单个单个的参数, 也可以直接传入一个对象, MyBatis 会根据对象属性的名称, 匹配到对应的 #{} 中.

同样, 如果对象使用了 @Param 进行参数绑定, 那么 #{} 中的参数名称要使用 对象名.属性名:

8. 删除数据(@Delete)

使用 MyBatis 删除记录, 使用 @Delete 注解. 

 


END 

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

大语言模型学习--本地部署DeepSeek

本地部署一个DeepSeek大语言模型 研究学习一下。 本地快速部署大模型的一个工具 先根据操作系统版本下载Ollama客户端 1.Ollama安装 ollama是一个开源的大型语言模型&#xff08;LLM&#xff09;本地化部署与管理工具&#xff0c;旨在简化在本地计算机上运行和管理大语言模型…

shell文本处理

shell文本处理 一、grep ​ 过滤来自一个文件或标准输入匹配模式内容。除了 grep 外&#xff0c;还有 egrep、fgrep。egrep 是 grep 的扩展&#xff0c;相当于 grep -E。fgrep 相当于 grep -f&#xff0c;用的比较少。 用法 grep [OPTION]... PATTERN [FILE]...支持的正则描述…

Linux中死锁问题的探讨

在 Linux 中&#xff0c;死锁&#xff08;Deadlock&#xff09; 是指多个进程或线程因为竞争资源而相互等待&#xff0c;导致所有相关进程或线程都无法继续执行的状态。死锁是一种严重的系统问题&#xff0c;会导致系统资源浪费&#xff0c;甚至系统崩溃。 死锁的定义 死锁是指…

Baklib内容中台赋能企业智管

内容中台构建全场景智管 现代企业数字化运营中&#xff0c;全域内容管理能力已成为核心竞争力。通过智能知识引擎驱动的内容中台架构&#xff0c;企业能够实现跨部门、多形态数据的统一归集与动态调度。以某制造企业为例&#xff0c;其利用中台系统将分散在CRM、ERP及内部文档…

ArcGIS Pro高级应用:高效生成TIN地形模型

一、引言 在地理信息科学与遥感技术的快速发展背景下&#xff0c;数字高程模型&#xff08;DEM&#xff09;已成为地形表达与分析的关键工具。 三角网&#xff08;TIN&#xff09;作为DEM的一种重要形式&#xff0c;因其能够精准描绘复杂地形特征而广受青睐。 ArcGIS Pro为用…

leetcode112-路径总和

leetcode 112 思路 我们利用递归来实现&#xff0c;用result字段来记录结果值&#xff0c;默认为false&#xff0c;我们递归的时候传入需要的目标值&#xff0c;然后每次遍历到一个节点&#xff0c;就用目标值减去节点当前值&#xff0c;最终到叶子节点时&#xff0c;如果是…

LLM | 论文精读 | CVPR | PEACE : 通过多模态大语言模型(MLLMs)赋能地质图全面理解

论文标题&#xff1a;FairCLIP: Harnessing Fairness in Vision-Language Learning 作者&#xff1a;Yan Luo Min Shi Muhammad Osama Khan Muhammad Muneeb Afzal等 期刊&#xff1a;CVPR 2025 email&#xff1a;yuhan.huangwhu.edu.cn 创作不易&#xff0c;恳请大家点赞收…

网络学习(四)HTTPS中,SSL的单向认证与双向认证

目录 一、什么是SSL&#xff1f;1.1 SSL 的主要功能1.2 SSL 的工作原理1.3 SSL 的核心组件1.4 SSL 的应用场景1.5 SSL 与 TLS 的区别 二、SSL 单向认证、双向认证2.1 SSL 单向认证2.2 SSL 双向认证2.3 总结&#xff1a;SSL 单向认证和双向认证的区别 一、什么是SSL&#xff1f;…

Mybatis 中#{} 和${} 的区别是什么?

在 MyBatis 中&#xff0c;#{} 和 ${} 都是用于动态 SQL 语句中的占位符&#xff0c;但是它们的作用和使用方式是不同的。下面是它们的区别&#xff1a; 1. #{} —— 用于防止 SQL 注入和自动类型处理 #{} 是用来将参数安全地传递到 SQL 语句中&#xff0c;它会将传递的参数值…

HTML-05NPM使用踩坑

2025-03-04-NPM使用踩坑 本文讲述了一个苦逼程序员在使用NPM的时候突然来了一记nmp login天雷&#xff0c;然后一番折腾之后&#xff0c;终究还是没有解决npm的问题&#x1f61e;&#x1f61e;&#x1f61e;,最终使用cnpm完美解决的故事。 文章目录 2025-03-04-NPM使用踩坑[toc…

Zookeeper 的核心引擎:深入解析 ZAB 协议

#作者&#xff1a;张桐瑞 文章目录 前言ZAB 协议算法崩溃恢复选票结构选票筛选消息广播 前言 ZooKeeper 最核心的作用就是保证分布式系统的数据一致性&#xff0c;而无论是处理来自客户端的会话请求时&#xff0c;还是集群 Leader 节点发生重新选举时&#xff0c;都会产生数据…

C++ Primer 动态数组

欢迎阅读我的 【CPrimer】专栏 专栏简介&#xff1a;本专栏主要面向C初学者&#xff0c;解释C的一些基本概念和基础语言特性&#xff0c;涉及C标准库的用法&#xff0c;面向对象特性&#xff0c;泛型特性高级用法。通过使用标准库中定义的抽象设施&#xff0c;使你更加适应高级…

基于 HTML、CSS 和 JavaScript 的智能九宫格图片分割系统

目录 1 前言 2 技术实现 2.1 HTML 结构 2.2 CSS 样式 2.3 JavaScript 交互 3 代码解析 3.1 HTML 部分 3.2 CSS 部分 3.3 JavaScript 部分 4 完整代码 5 运行结果 6 总结 6.1 系统特点 6.2 使用方法 1 前言 在当今数字化的时代&#xff0c;图片处理需求日益增长。…

Java+iTextPDF,实时生成与预览PDF文件的最佳实践!

Java+iTextPDF,实时生成与预览PDF文件的最佳实践! 背景 其实公司之前的项目里是用到了帆软报表的,然而最近接了一个新项目,这个项目独立部署在甲方的独立环境中,组长的意思是不用再单独部署一套帆软报表,成本太大,用其他方式实现一下。虽然我不太理解成本大在哪儿,不…

Linux 快捷命令链接

修改mvn命令 默认手动安装后&#xff0c;命令格式为 安装路径命令 /data/apache-maven-3.8.8/bin/mvn -v更改为通用的命令模式 [root ~]# cat /etc/centos-release CentOS Linux release 7.9.2009 (Core) [root ~]# echo $PATH /usr/local/node16/bin:/usr/local/sbin:/sbin…

论文回顾:NeoBERT:新一代 BERT

NeoBERT&#xff1a;新一代编码器&#xff0c;具有 4K 标记上下文长度&#xff0c;在 MTEB 上优于 RoBERTa 等更大的模型&#xff01; 论文链接&#xff1a;https://arxiv.org/pdf/2502.19587 摘要 NeoBERT 是下一代双向编码器&#xff1b;它融合了最先进的架构、现代数据和优…

机器视觉开发教程——封装Halcon通用模板匹配工具【含免费教程源码】

目录 引言前期准备Step1 设计可序列化的输入输出集合【不支持多线程】Step2 设计程序框架1、抽象层【IProcess】2、父类【HAlgorithm】3、子类【HFindModelTool】 Step3 设计UI结果展示 引言 通过仿照VisionPro软件二次开发Halcon的模板匹配工具&#xff0c;便于在客户端软件中…

一、OpenGL的原理解析

文章目录 OpenGL到底实现的是什么&#xff1f;OpenGL内模型数据的本质是什么&#xff1f;为什么三角形是 3D 渲染的最基本单元&#xff1f;MVP 变换&#xff08;Model-View-Projection 变换&#xff09;OpenGL渲染流程-摄像机变换OpenGL渲染流程-投影变换OpenGL渲染管线概述 Op…

大模型——CogView4:生成中英双语高清图片的开源文生图模型综合介绍

CogView4:生成中英双语高清图片的开源文生图模型综合介绍 CogView4 是由清华大学 KEG 实验室(THUDM)开发的一款开源文生图模型,专注于将文本描述转化为高质量图像。它支持中英双语提示词输入,尤其擅长理解中文提示并生成带有汉字的图像,非常适合广告设计、短视频创作等场…

网络安全法与等级保护 PPT 精华汇总

资源描述 本资源文件为《网络安全法与等级保护》的PPT精华汇总&#xff0c;内容涵盖了网络安全法与等级保护的总体框架及相关标准规范。该PPT详细介绍了网络安全法与等级保护的各个章节和条款&#xff0c;并提供了基础类和应用类的相关标准文件&#xff0c;帮助读者全面了解和…