【MyBatis】一文教你学会使用MyBatis操作数据库

news2025/1/22 6:21:24

文章目录

  • 1. MyBatis是什么?
  • 2. 为什么要学MaBatis?
  • 3. MyBatis环境搭建
  • 4. MyBatis的使用
    • 4.1 简单示例
    • 4.2 获取动态参数
      • 4.2.1 ${xxx}获取动态参数
      • 4.2.2 #{xxx}获取动态参数
      • 4.2.3 #{xxx}与${xxx}获取字符串类型数据
      • 4.2.4 sql注入问题
      • 4.2.5 模糊查询like
      • 4.2.6 #{}与${}区别总结
  • 5. 修改操作(增删改)
  • 6. 查询操作
    • 6.1 resultType与resultMap
    • 6.2 单表查询
    • 6.3 多表查询
  • 7. 动态SQL
    • 7.1 \<if>标签
    • 7.2 \<trim>标签
    • 7.3 \<where>标签
    • 7.4 \<set>标签
    • 7.5 \<foreach>标签

1. MyBatis是什么?

MyBatis是一款持久层框架,它可更简单的完成程序与数据库之间交互的操作,它可以更简单的操作与读取数据库,它还支持自定义SQL,存储过程及高级映射

2. 为什么要学MaBatis?

对于后端开发来说,完整的程序由以下两部分组成:

  • 后端程序
  • 数据库

在这里插入图片描述

后端程序与数据库之间的通讯,就得依靠数据库连接工具,像之前学习的JDBC,但是因为JDBC使用起来太繁琐了,简单回顾使用JDBC操作数据库的步骤:

  1. 创建数据库连接池DataSource
  2. 使用DataSource获取数据库连接Connection
  3. 编写预编译的sql
  4. 使用Connection及sql创建操作命令对象Statement
  5. 替换占位符
  6. 使用Statement执行sql
  7. 如果是查询,则返回并处理结果集
  8. 释放资源

所以呢,使用MaBatis简化上述操作,方便快速的操作数据库

3. MyBatis环境搭建

创建Spring Boot项目的时候,添加MySQL驱动依赖与MyBatis框架依赖

在这里插入图片描述

创建好项目后,如果直接启动项目会报错,因为启动项目时候要连接数据库,此时我们还没有配置数据库源

在配置文件application.properties中配置数据库连接信息

# 配置数据库连接信息
spring.datasource.url=jdbc:mysql://localhost:3306/blog?characterEncoding=utf8&useSSL=false
spring.datasource.username=root
spring.datasource.password=xiaobai520..@@@
# 配置数据库驱动
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver

在配置文件application.properties中配置MyBatis的xml文件路径

# 配置mybatis xml路径
# 在根路径下有mybatis文件夹,里面有xxxMapper.xml文件
mybatis.mapper-locations=classpath:/mybatis/*Mapper.xml

配置MyBatis中sql执行打印

#配置mybatis sql执行打印
mybatis.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl

# sql打印日志的级别为debug,但是spring boot默认的打印级别为info
# 设置com.example.demo包下的打印级别为debug
logging.level.com.example.demo=debug

4. MyBatis的使用

4.1 简单示例

我们按照下面的流程来完成MyBatis操作数据库的功能

在这里插入图片描述

MyBatis模式开发由两部分组成:

  1. interface:让其他层可以注入使用的接口
  2. xxx.xml:具体的sql(可认为它是interface的实现)

具体步骤

创建Mapper接口
在这里插入图片描述
创建对应的xml文件
在这里插入图片描述
在xml中编写sql
在这里插入图片描述
创建service类调用接口中的方法

在这里插入图片描述
在controller类中调用service方法
在这里插入图片描述
启动项目,在浏览器输入访问地址,可以看到返回一条数据

在这里插入图片描述

4.2 获取动态参数

4.2.1 ${xxx}获取动态参数

上述的示例中,查询所有用户的信息,那此时如果想获得id为1的用户的信息该怎么操作?那就要在接口的方法中传递参数了

在这里插入图片描述
@Param与@RequestParam用法差不多,也是获取并赋值的意思,并且要求必传,也就是要必须能拿到@Param中的键对应的值

此时xml中的sql应该这样写:

在这里插入图片描述
使用${xxx}表示传递的参数,此时xxx必须与@Param中的键对应,如果接口方法中没有用@Param修饰形参,那xxx必须与形参名称对应

在测试类中运行结果如下:

在这里插入图片描述
在这里插入图片描述
发现使用${}获取动态参数会直接替换

4.2.2 #{xxx}获取动态参数

将sql改为使用#{xxx}的方式替换

在这里插入图片描述

再次执行测试类看打印结果:

在这里插入图片描述
在这里插入图片描述
发现使用#{}获取动态参数是采用占位符也就是预编译的方式替换

4.2.3 #{xxx}与${xxx}获取字符串类型数据

从上述的示例,获取数据库字段类型为int来看,使用这两种方式没有什么区别

此时使用名称获取用户数据,也就是数据库中字段类型为varchar,先使用#{}获取参数
在这里插入图片描述
启动测试,观察打印结果:发现正确获取到用户信息

在这里插入图片描述

将获取参数方式改为${},再观察打印结果:发现此时报错
在这里插入图片描述
原因:${}方式是直接替换,此时sql为select * from user where username=admin,但是正确的sql应该为select * from user where username=‘admin’,程序中执行的sql,admin没有带引号

如果使用KaTeX parse error: Double superscript at position 21: …数的话,我们可以手动加上`' '̲`,再观察打印情况 ![在这里…{}的话,会存在sql注入问题

4.2.4 sql注入问题

模拟登录,因为sql注入一般发生在登录阶段,使用${}获取参数
在这里插入图片描述

测试类,使用正确的用户名和密码:可正确获取用户数据

在这里插入图片描述

在这里插入图片描述

测试类,将用户密码改为错误的,观察打印结果:获取不到用户数据
在这里插入图片描述

测试类,使用错误的用户密码 ' or 1='1,观察打印结果:

在这里插入图片描述

发现尽管输入错误的密码,也是可以获取到用户数据,这就是典型的sql注入问题

我们再使用#{}方式获取参数,看看是否也可以获取到数据
在这里插入图片描述
发现使用错误的密码获取不到数据,也就是使用#{}不存在sql注入问题

4.2.5 模糊查询like

使用#{}进行模糊匹配
在这里插入图片描述

发现程序会报错,因为使用#{},sql变为select * from user where username like '%'a'%',因为#{}使用预编译的方式,对于字符串类型会加上一层引号,所以导致sql错误

此时可以使用concat拼接函数改造sql
在这里插入图片描述
在这里插入图片描述
发现如果对于like模糊查询,使用#{}的时候得结合concat()函数

除此外,我们可以使用${},因为${}是直接替换sql,所以不存在问题

在这里插入图片描述

在这里插入图片描述

4.2.6 #{}与${}区别总结

  • ${}是将参数直接替换到sql中,#{}是使用占位符也就是预编译的方式
  • 当参数为数值类型时,两种方式没区别
  • 当参数为字符串类型时,#{}使用占位符预执行,${}直接替换为sql,不加单引号,所以${}存在sql注入问题
  • 当参数为sql本身的时候,使用${}

5. 修改操作(增删改)

增加insert

在这里插入图片描述
打印结果:
在这里插入图片描述
观察数据库是否真正插入:发现插入数据库数据成功
在这里插入图片描述
我们在测试类中的方法默认是会持久化到数据库的,如果在方法加上 @Transactional,表示开启事务,则在方法执行结束会进行回滚操作,那么此次对数据库的操作将不会持久化

上述例子表示默认情况返回的是受影响的行数,但是我们增加一条数据时,想返回主键id,则可以按照如下方式进行操作
在这里插入图片描述
测试方法如下
在这里插入图片描述
打印结果:
在这里插入图片描述

  • useGeneratedKeys:开启还是关闭取出由数据库默认生成的主键,默认是false
  • keyProperty:唯一标识对象的属性,useGeneratedKeys获取的主键值赋值给keyProperty设置的属性
  • keyColumn:设置⽣成键值在表中的列名,如果⽣成列不⽌⼀个,可以⽤逗号分隔多个属性名称

删除delete

在这里插入图片描述

更新update

在这里插入图片描述

6. 查询操作

6.1 resultType与resultMap

resutType

我们上述的例子中查询操作都使用的是resultType,它是默认让数据库字段名与实体类属性名做了映射,也就是字段名与属性名必须对应起来,如果某个属性和字段没有对应,那么该字段的值就不会映射到属性上,导致该属性的值是一个null
在这里插入图片描述
如果字段与属性不对应的话,还想继续使用resultType的话,我们在可以使用as对字段起别名
在这里插入图片描述

resultMap

有时候程序中对象的属性和数据库的字段对应不起来,我们可以使用resultMap做映射

如下所示,user的属性pwd与数据库的password对应不起来,此时可以使用resultMap
在这里插入图片描述
在这里插入图片描述

6.2 单表查询

前面涉及到的查询都是单表查询,此处不过多介绍

6.3 多表查询

现在有两张表,user和article
在这里插入图片描述
我们现在如果要查询某篇文章,并且要查询出它的用户信息,也就是uid对应到user中id的用户的用户名(username)信息,此时发现我们的两个实体类User,Article都不能对应完成此次查询的所有信息
在这里插入图片描述

此时我们可以创建一个包vo,在此包下创建一个类ArticleVO,此类继承Article,并且添加一个属性username,此时该类就可完成我们此处的多表查询操作

在这里插入图片描述

@Data
public class ArticleVO extends Article {
    private String username;

    @Override 
    public String toString() {
        return "ArticleVO{" +
                "username='" + username + '\'' +
                '}'+super.toString();
    }
}

加了@Data注解为什么还要重写toString()?

因为@Data注解的toString方法只给该类的一个属性username输入打印,但是此类继承了Article,我们希望toString可以可以打印出继承的父类属性信息

创建ArticleMapper接口及ArticleMapper.xml文件

在这里插入图片描述
编写测试类及测试方法
在这里插入图片描述
启动观察结果:获取到我们想要的数据
在这里插入图片描述

7. 动态SQL

动态sql是MyBatis的强大特性之一,它可以完成不同条件下不同的sql拼接,它是在xml文件中进行不同条件拼接的

7.1 <if>标签

比如我们插入用户信息,有username(用户名),password(密码),photo(图像),但是当传递的用户信息中photo为null时,我们就不插入photo信息,如果用户信息中photo不为null,我们就插入photo信息

photo信息为null

在这里插入图片描述
说明: if标签中test属性中的photo为传递对象的属性而不是数据库表的字段

测试方法及打印结果:
在这里插入图片描述

photo信息不为null

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

7.2 <trim>标签

上述的<if>标签如果使用了多个,如下面这种情况
在这里插入图片描述
为了解决上述问题,我们在整个需要判断条件的语块中添加<trim>标签

关于<trim>标签的属性:

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

注意: 当trim标签里没有任何内容的时候,是不会加上前缀后缀的,只有trim里面有内容,前缀和后缀的设置才生效

我们可以调整上述xml中的sql为:

在这里插入图片描述

测试方法:

在这里插入图片描述

打印结果:

在这里插入图片描述

7.3 <where>标签

当传入对象,根据属性进行where条件筛选的时候,当属性不为null的都为查询条件

当<where>标签中有内容时候,此时sql会加上where进行条件筛选,如果无内容则不会添加where,并且<where>标签会去除前缀and(不能去掉后缀and

在这里插入图片描述

7.4 <set>标签

根据传入的对象属性来更新数据,如果<set>标签里没内容,则不生成set,当内容不为空的时候会生成set,并且<set>标签会自动去除后缀 ,不会去除前缀

    <update id="updateById">
        update user
        <set>
            <if test="username != null">
                username=#{username}
            </if>
            <if test="password != null">
                password=#{password}
            </if>
        </set>
        <where>
            <if test="id != null">
                id=#{id}
            </if>
        </where>
    </update>

7.5 <foreach>标签

对某个集合遍历可以使用该标签,该标签有如下属性:

  • collection:绑定方法中的集合对象
  • item:遍历时,集合中的每一个元素
  • open:给语句块添加前缀
  • close:给语句块添加后缀
  • separator:间隔字符串

示例:我们删除id为6,7,8,9的用户

在这里插入图片描述

测试方法:

    @Test
    void deleteByIds() {
        int[] ids = {6,7,8,9};
        int n = userMapper.deleteByIds(ids);
        System.out.println(n);
    }

打印结果:
在这里插入图片描述

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

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

相关文章

动态规划-背包问题-背包九讲

title: 动态规划-背包问题 date: 2023-05-12 13:01:31 categories: Algorithm动态规划 tags:动态规划 01背包问题 题目 题目链接&#xff1a;https://www.acwing.com/problem/content/description/2/ 有 N N N 件物品和一个容量是 V V V 的背包。每件物品只能使用一次。 …

数据分析05——往Pandas中导入数据

1、导入Excel&#xff1a; 注意这种方法可以导入xlsx和xls两种类型的数据读入的数据会以DataFrame的格式显示举例&#xff1a;df pd.read_excel(‘./data/demo_03.xlsx’)还可以导入excel文件中具体的某一个表格&#xff1a;pd.read_excel(‘./data/demo_03.xlsx’, sheet_na…

SD-MTSP:麻雀搜索算法SSA求解单仓库多旅行商问题(提供MATLAB代码,可以修改旅行商个数及起点)

一、单仓库多旅行商问题 多旅行商问题&#xff08;Multiple Traveling Salesman Problem, MTSP&#xff09;是著名的旅行商问题&#xff08;Traveling Salesman Problem, TSP&#xff09;的延伸&#xff0c;多旅行商问题定义为&#xff1a;给定一个&#x1d45b;座城市的城市集…

postgresql源码学习(55)—— 列中的NULL值是如何存储和判断的?

问题来自 《PostgreSQL面试题集锦》学习与回答_Hehuyi_In的博客-CSDN博客 第11题 一、 NULL值存储位置 在pg元组头数据中&#xff0c;有一个t_bits数组&#xff0c;用于存储空值位图。当元组中没有null值的时候&#xff0c;t_bits可以被认为是空的&#xff0c;当元组有null值的…

javaweb学习 html+css基础1

1、学习路线 2、web网站的开发模式 课程安排 web 前端 前端学习内容 html和css的帮助网站 HTML 系列教程 (w3school.com.cn) html快速入门 使用VSCode开发工具&#xff0c;输入 &#xff01;enter&#xff0c;即可出现骨架。 ctrl/ 可以将文字转换为注释 右击一个页面…

【JavaEE】计网之IP协议+以太网+DNS

博主简介&#xff1a;想进大厂的打工人博主主页&#xff1a;xyk:所属专栏: JavaEE初阶 网络层重点协议——IP协议&#xff0c;在复杂的网络环境中确定一个合适的路径~ 本篇文章我们来讲解什么是 IP协议及以太网&#xff0c;在IP协议中&#xff0c;有 地址管理 和 路由选择 两个…

C++中基类和派生类的构造函数与析构函数的调用顺序分析

基类为B&#xff0c;派生类为C&#xff0c;直接上代码以及运行结果。 目录 思路分析 代码一 运行结果一 代码二&#xff1a;B(i)改为b(i) 运行结果二 代码三&#xff1a;加上B(i) 运行结果三 代码四&#xff1a;删掉C类定义的B对象b&#xff0c;删除b(i) 运行结果四 思路…

Linux使用全应用

一、CentOS安装Docker Docker CE 支持 64 位版本 CentOS 7&#xff0c;并且要求内核版本不低于 3.10&#xff0c; CentOS 7 满足最低内核的要求&#xff0c;所以我们在CentOS 7安装Docker。 基础命令 搜索镜像&#xff1a;docker search mysql 下载镜像&#xff1a;docker p…

[架构之路-197]-《软考-系统分析师》- 关键技术 - 问题分析阶段重要的四个任务

目录 前言&#xff1a; 一、信息系统/软件产品的问题分析概述 二、信息系统/软件产品的问题/痛点分析四步骤 步骤1、问题领域分析&#xff1a;研究遇到问题的业务领域&#xff08;诉求&#xff09; 步骤2、 领域问题和机会分析&#xff1a;分析业务领域的问题、痛点、难点…

计算机网络学习 一 (计算机网络体系结构)

计算机网络 基本概念 计算机网络是一个将分散的,具有独立功能的计算机系统. 简单来说,计算机网络就是一些互连的,自治的计算机系统的集合 分类 广义上:是一个资源共享的系统. 资源共享上: 1.目的–资源共享 2.组成单元–分布在不同地理位置的多台独立的"自治计算机"…

第十四届蓝桥杯青少组模拟赛Python真题 (2022年11月8日)

第十四届蓝桥杯青少组模拟赛Python真题 (2022年11月8日) 编程题 第 1 题 问答题 二进制位数 十进制整数2在十进制中是1位数&#xff0c;在二进制中对应10&#xff0c;是2位数。 十进制整数22在十进制中是2位数&#xff0c;在二进制中对应10110&#xff0c;是5位数。 请问十…

Linux高级---k8s之service服务

文章目录 一、service基本概念二、service类型三、service的使用1、实验环境准备2、ClusterIP类型的Service3、HeadLiness类型的Service4、NodePort类型的Service5、LoadBalancer类型的Service6、ExternalName类型的Service 一、service基本概念 在kubernetes中&#xff0c;pod…

【Java多线程编程】线程的六种状态

前言&#xff1a; 在我们进行多线程编程&#xff0c;脑海里会想到线程运行的状态到底是什么&#xff1f;因此我整理出这线程的状态这篇博文。线程的状态分为六种&#xff1a;新建状态&#xff08;NEW&#xff09;、就绪状态&#xff08;RUNNABLE&#xff09;、阻塞状态&#xf…

1-Linux环境安装JDK

Linux环境安装JDK 准备&#xff1a; ① Linux 环境 本文中Linux环境为 CentOS Linux 7 可使用以下命令查询 linux 系统版本&#xff1a; hostnamectl② 准备JDK包 进入官网 https://www.oracle.com/java/technologies/downloads/#java17下载对应jdk包 此处使用以前下载的旧…

Linux驱动编程(驱动程序基石)(下)

一、中断的线程化处理 复杂、耗时的事情&#xff0c;尽量使用内核线程来处理。上节视频介绍的工作队列用起来挺简单&#xff0c;但是它有一个缺点&#xff1a;工作队列中有多个 work&#xff0c;前一个 work 没处理完会影响后面的 work。解决方法有很多种&#xff0c;比如干脆…

String类的学习笔记(下):字符串拼接以及StringBuilder和StringBuffer的学习

本文介绍了String类对字符串进行拼接的方法 和拼接字符串的效率分析 以及能对字符串内容进行修改的StringBuilder和StringBuffer类其常用方法和区别 , 最后介绍了两个字符串经典面试题 StringBuilder和StringBuffer的学习 一.String类概括二.StringBuilder和StringBuffer1.字符…

是未来的超级计算机还是只是一场炒作?

随着科技的飞速发展和创新&#xff0c;量子计算技术逐渐成为了人们关注的热点话题。量子计算作为一种前沿的计算方式&#xff0c;具有超强的运算能力和突破性的创新潜力&#xff0c;因此备受瞩目。然而&#xff0c;随着各大公司和机构纷纷加入到这一领域的竞争中&#xff0c;一…

超详细github配置(仔细看,看完不会,你怪我)

github的重要性&#xff1a; 网络时代的程序员必备。 github的作用&#xff1a; 版本管理多人协作开源共享 常用方案&#xff1a; gitTortoiseGitgithub [Tortoise&#xff0c;程序员常称其为小乌龟&#xff0c;小海龟] 安装配置步骤 1.注册 GitHub: Where the world bu…

服务(第二十二篇)主从复制和读写分离

主从复制原理&#xff1a; 首先主节点会开启二进制日志&#xff0c;从节点会开启中继日志&#xff0c;从节点会开启io线程检测主节点是否有更新&#xff0c;如果更新了就会向主节点请求二进制事件&#xff0c;主会开启dump线程发送二进制事件&#xff0c;然后保存在从节点的中…

假如面试官让你十分钟完成双向循环链表

&#x1f48c; 博客内容&#xff1a;假如面试官让你十分钟完成双向循环链表&#xff0c;多一秒都不行 &#x1f600; 作  者&#xff1a;陈大大陈 &#x1f680; 个人简介&#xff1a;一个正在努力学技术的准前端&#xff0c;专注基础和实战分享 &#xff0c;欢迎私信&…