MySQL-事务Transaction详解

文章目录

  • 事务概述
    • 事务基本概念
    • 事务四大特性(ACID)
    • 演示MySQL事务
      • 手动开启事务
      • MySQL默认事务机制
  • 事务的隔离级别
    • 隔离级别基本概述
    • 三种现象
      • 脏读
      • 不可重复读
      • 幻读
    • 查看和设置隔离级别
    • 四种隔离级别及演示
      • 读未提交(read uncommitted)
      • 读提交(read committed)
      • 可重复读(repeatable read)
      • 串行化(serializable)
    • 可重复读的幻读问题

事务概述

事务基本概念

  1. 事务是最小的一个工作单元, 在数据库当中, 事务表示一件完整的事
  2. 一个业务的完成可能需要多条DML语句共同配合才能完成, 例如转账业务, 需要执行两条DML语句, 先更新张三的业务, 再更新李四的业务, 为了保证转账业务不出现问题, 就必须保证这两条DML语句必须同时成功或者同时失败, 所以我们就需要借助业务这一机制来完成
  3. 也就是说使用了事务机制之后, 在同一个事务内部, 多条DML语句会同时成功或者同时失败, 不会出现一部分成功一部分失败的现象
  4. 事务只针对DML语句(insert, delete, update)有效, 因为只有这三个语句是改变表中数据
  5. 事务只针对DML语句有效, 但是在一个事务内部是可以有DQL语句进行结果查询的

事务四大特性(ACID)

  1. 原子性(Atomicity)
    是指事务包含的所有操作要么全部成功,要么同时失败
  2. 一致性(Consistency)
    是指事务开始前,和事务完成后,数据应该是一致的。例如张三和李四的钱加起来是5000,中间不管进行过多少次的转账 操作(update),总量5000是不会变的。这就是事务的一致性。
  3. 隔离性(Isolation)
    隔离性是当多个⽤户并发访问数据库时,⽐如操作同⼀张表时,数据库为每⼀个⽤户开启的事务,不能被其他事务的操作所⼲扰,多个并发事务之间要相互隔离, 隔离性主要解决的是多个事务并发的情况, 这在下面的内容里很重要
  4. 持久性(Durability)
    持久性是指⼀个事务⼀旦被提交了,那么对数据库中的数据的改变就是永久性的,即便是在数据库系统遇到故障的情况下也不会丢失提交事务的操作

演示MySQL事务

手动开启事务

在dos命令行窗口的基本操作指令

事务操作基本指令
开启事务start transaction / begin
提交事务(成功提交事务)commit
回滚事务(未成功提交事务)rollback

只要执行上面的commit或者是rollback事务都会结束, 前者是成功提交操作(即所有的操作都生效), 后者是回滚(未成功提交)撤销事务中的所有操作
演示如下
首先是查找一下t_student表中的数据
在这里插入图片描述
现在我们开启事务并执行一些DML操作后commit(成功提交)
在这里插入图片描述
我们发现成功插入了一条数据
现在我们再次开启一个事务并执行一些DML操作后rollback(未成功提交)
在这里插入图片描述
此时我们的stu_no = 13的这行记录并没有被删除…

MySQL默认事务机制

关于MySQL默认的事务机制:
MySQL默认情况下采用的事务机制是 : 自动提交, 所谓自动提交就是只要执行一条DML语句, 就会自动的开启事务并且执行结束之后直接commit, 所以这时候我们rollback是无法进行回滚的…(ACID中的持久性)

测试如下
在这里插入图片描述
分析一下底层的运行机制, 当执行insert语句的时候, 底层自动开启了事务机制, 这条SQL执行结束之后默认进行commit操作, 所以此时rollback并不能使之前的事务回滚

事务的隔离级别

隔离级别基本概述

事物的隔离级别这一块主要针对的就是上面事务四大特性中的隔离性(Isolation)
隔离性感性的理解可以视为两个不同事务之间的"墙", 而隔离级别就是墙的厚度大小

隔离级别脏读不可重复读幻读
读未提交(read uncommitted)存在存在存在
读提交(read committed)不存在存在存在
可重复读(repeatable read)不存在不存在存在(大部分可避免)
串行化(serializable)不存在不存在不存在

事务的隔离级别分为四种(按照隔离程度从低到高)
读未提交 < 读提交 < 可重复读 < 串行化
不同隔离级别产生的现象(按照严重程度从低到高)
脏读 > 不可重复读 > 幻读

三种现象

脏读

官方术语

指的是一个事务读取了另一个事务尚未提交的数据,即读取了另一个事务中的脏数据(Dirty Data)。在此情况下,如果另一个事务回滚了或者修改了这些数据,那么读取这些脏数据的事务所处理的数据就是不准确的。

简单点说就是, 现在假设开启了A, B两个事务, B事务对其中的数据进行了DML语句操作(增删改), B事务并没有提交,此时我们在A事务的内部进行select语句查询, 居然能读取到被B事务增删改的数据, 是不是就说明A, B两个事务之间的隔离级别非常弱(两个事务之间的"墙"厚度很薄)

不可重复读

官方术语
注意, 这里所说的现象是针对同一条SQL来说的, 也就是同一条SQL的结果不一样

指在一个事务内,多次读取同一个数据行,得到的结果可能是不一样的。这是由于其他事务对数据行做出了修改操作,导致数据的不一致性。

翻译一下, 现在假设我们开启了A, B两个事务, 现在在A事务中执行一条DQL查询语句, 此时我们在B事务中使用DML语句中的update语句进行操作A中查询的区间内的数据, B是否提交未知, 我们此时在A中再次执行之前那条DQL语句, 会发现我们前后的数据是不一样的, 为什么说B是否提交未知呢, 因为如果是B执行之前进行select查询, 这种情况就属于脏读(也是不可重复读), 所以说脏读的严重性要大于不可重复读

幻读

官方术语

指在事务执行过程中,前后两次相同的查询条件得到的结果集不一致,可能会变多或变少。

其实就是A, B两个事务开启之后, B中添加几条/删除几条数据, A中可以观测到
关于不可重复度和幻读的区别, 不可重复读是指的是一条数据的内部更新变化, 而幻读指的是添加或者删除某些数据

查看和设置隔离级别

MySQL默认的隔离级别是可重复读(repeatable read)

查看隔离级别(全局/对话)

-- 查看当前会话的隔离级别
-- 也就是当前dos窗口的隔离级别(退出窗口之后失效)
select @@transaction_isolation;
-- 查看当前全局的隔离级别(整个MySQL服务)
select @@global.transaction_isolation;

演示如下
查看当前对话的
在这里插入图片描述
查看全局的
在这里插入图片描述
设置隔离级别
当设置全局的隔离级别之后, 重新打开对话之后对话的隔离级别也会随之改变

-- 设置当前对话的隔离级别
set session transaction isolation level [级别名称]
-- 设置全局的隔离级别
set global transaction isolation level [隔离名称]

下面演示的是把当前对话的隔离级别改成读提交
在这里插入图片描述
这种设置的作用范围就是当前的对话, 一旦对话撤销之后也随之失效

下面演示的是设置全局的隔离级别为读提交
在这里插入图片描述
这种全局的隔离级别一旦设置, 全局范围内都有效, 不能随意切换

四种隔离级别及演示

我们对隔离级别的模拟是通过开启两个dos窗口来进行模拟并发的

读未提交(read uncommitted)

A事务与B事务,A事务可以读取到B事务未提交的数据。这是最低的隔离级别。几乎两个事务之间没有隔离。这种隔离级别是一种理论层面的,在实际的数据库产品中,没有从这个级别起步的。当事务隔离级别是读未提交时,三种现象都存在:脏读,不可重复读,幻读
首先设置一下全局的隔离级别

-- 设置全局的隔离级别为读未提交
set global transaction isolation level read uncommitted;

下面是演示
我们使用的数据演示表是下面的学生表, 这个是初始的数据情况
在这里插入图片描述

事务A事务B
开启事务A在这里插入图片描述开启事务B在这里插入图片描述
插入一条数据在这里插入图片描述
更改一条数据在这里插入图片描述
再次查询一下数据在这里插入图片描述
rollback回滚B事务

通过对比我们很明显的看到, 在B事务没有提交的情况下, 我们A事务就可以查看到B事务中新插入的这条信息的数据(幻读), 也可以查询到B事务中对A事务中信息的更新(不可重复读), 这两个都同时属于脏读, 可见此时的隔离级别是非常低的

读提交(read committed)

A事务与B事务,A事务可以读取到B事务提交之后的数据。Oracle数据库默认的就是这种隔离级别
此时存在不可重复读和幻读
首先设置一下全局的隔离级别为读提交

-- 设置全局的隔离级别为读提交
set global transaction isolation level read committed;

下面是演示

事务A事务B
开启事务A在这里插入图片描述开启事务B在这里插入图片描述
查询一下数据在这里插入图片描述
插入一条数据在这里插入图片描述
更新一条数据在这里插入图片描述
在B事务未提交之前查一下数据在这里插入图片描述
提交B事务在这里插入图片描述
重新查一下A事务数据在这里插入图片描述
提交A事务在这里插入图片描述

分析一下上面的过程, 在B事务没有提前之前, 我们在B中执行的更新删除的操作都不会影响事务A中的查询结果, 也就是避免了脏读, 但是当提交了B事务中的数据, A事务就可以查找出来B事务增加和修改的数据, 也就是存在不可重复读和幻读

可重复读(repeatable read)

这个隔离级别是MySQL数据库默认的。A事务和B事务,A事务开启后,读取了某一条记录,然后B事务对这条记录进行修改并提交,A事务读取到的还是修改前的数据。这种隔离级别称为可重复读。
这个隔离级别可以避开脏读, 不可重复读, 大部分的幻读(待会再说)
首先还是修改一下当前的隔离级别

-- 设置全局的隔离级别为可重复读
set global transaction isolation level repeatable read;

下面是演示

A事务B事务
开启A事务在这里插入图片描述开启事务B在这里插入图片描述
查询一下原始数据在这里插入图片描述
插入一条数据在这里插入图片描述
更新一条数据在这里插入图片描述
再次查看数据在这里插入图片描述
提交事务在这里插入图片描述
再次查看数据在这里插入图片描述
使用for update查询在这里插入图片描述
提交事务在这里插入图片描述

经过上面的演示, 我们发现在B事务提交之前, B事务中执行的任何操作A事务中都无法查询到, 所以这就避免了脏读, 此时我们提交事务B, 再次对A事务中的事务尽心查询, 我们发现这里不仅避免了不可重复读(同一条数据都是一样的), 同时甚至避免了幻读(数据的条数并没有增加), 此时我们使用了for update查询(这是查询语句可以读取到当前最新的数据), 就会发现产生了幻读, 所以我们说MySQL可以最大程度的避免幻读但是也不能完全避免, 至于什么时候可以, 什么时候不可以我们下面会介绍, 还有人疑问这里难道不是不可重复读么, 注意此时select语句发生了变化, 所以此时还是满足可重复读的

串行化(serializable)

这种隔离级别最高,避免了所有的问题,缺点是效率低,因为这种隔离级别会导致事务排队处理,不支持并发。所以我们一般不会采用这种隔离级别
首先还是设置一下全局的隔离级别

set global transaction isolation level serializable;

下面是演示

事务A事务B
开启A事务在这里插入图片描述开启B事务在这里插入图片描述
事务B进行操作在这里插入图片描述
提交事务在这里插入图片描述
此时B事务才可以执行在这里插入图片描述

分析一下, 此时隔离级别为串行化的级别, 此时不允许并发了, 我们发现我们在B事务(后开启)中执行DQL查询操作, 光标就会一直闪烁, 卡在这里不执行, 等待A事务提交完毕之后才可以执行, 当A事务提交之后, B事务自动执行, (45.89 sec 就是B事务等待的秒数), 所以这种情况我们很少用, 不支持并发之后效率很低

可重复读的幻读问题

上面我们说了, MySQL中的可重复读可以尽自己最大的努力来避免产生幻读问题, 但是还是可能存在的, 下面我们详细的分析一下可重复度隔离级别中的幻读问题

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

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

相关文章

颠覆Transformer的Mamba模型[精简版本]------S4

1、改进transformer不擅长处理超长的序列的问题:输入u到状态x 序列数据一般都是离散的数据 比如文本、图、DNA,但现实生活中还有很多连续的数据,比如音频、视频,对于音视频这种信号而言,其一个重要特点就是有极长的context window,而在transformer长context上往往会失败,…

Spring Boot技术栈的电影评论网站设计与实现

6系统测试 6.1概念和意义 测试的定义&#xff1a;程序测试是为了发现错误而执行程序的过程。测试(Testing)的任务与目的可以描述为&#xff1a; 目的&#xff1a;发现程序的错误&#xff1b; 任务&#xff1a;通过在计算机上执行程序&#xff0c;暴露程序中潜在的错误。 另一个…

算法——python实现堆排序

文章目录 堆排序二叉树堆堆排序的过程&#xff1a;代码实现python中的heapq模块 堆排序 二叉树 关于二叉树的操作&#xff0c;其实核心就是 父节点找子节点&#xff0c;子节点找父节点 如果要将二叉树存储到队列中&#xff0c;就需要找出 父子节点之间的规律&#xff1a; 父…

什么是SYN flood,如何处理

在数字化时代&#xff0c;随着互联网的普及和技术的飞速发展&#xff0c;网络安全问题变得日益严峻。Flood攻击&#xff0c;作为一种典型的网络攻击手段&#xff0c;对个人和企业的信息安全构成了重大威胁。通过深入了解Flood攻击的概念、特点、影响及解决方案&#xff0c;我们…

Sentinel 快速入门

前置推荐阅读:Sentinel 介绍-CSDN博客 前置推荐阅读&#xff1a;Nacos快速入门-CSDN博客 快速开始 欢迎来到 Sentinel 的世界&#xff01;这篇新手指南将指引您快速入门 Sentinel。 Sentinel 的使用可以分为两个部分: 核心库&#xff08;Java 客户端&#xff09;&#xff1a…

现代数字信号处理I-P4 CRLB+LMMSE 学习笔记

目录 学习资料视频链接&#xff1a; 1. 估计参数的CRLB回顾 2. 参数变换下的CRLB拓展 3. 矢量参数下的CRLB扩展 3.1 矢量参数下的CRLB公式 3.2 两个矩阵不等式关系的意义说明 3.3 矢量参数下CRLB公式的证明过程 4. 线性估计 重点注意事项&#xff1a;此处的线性估计&am…

【React】React18核心源码解读

前言 本文使用 React18.2.0 的源码&#xff0c;如果想回退到某一版本执行git checkout tags/v18.2.0即可。如果打开源码发现js文件报ts类型错误请看本人另一篇文章&#xff1a;VsCode查看React源码全是类型报错如何解决。 阅读源码的过程&#xff1a; 下载源码 观察 package…

【java面经thinking】二

目录 redis了解 使用原因 应用场景 数据类型 redis事务 数据持久化 RDB(快照)&#xff1a; AOF(即时更新)&#xff1a; 选择方式&#xff1a; redis快速的原因 redis单线程 单机瓶颈 经典3问 参考博客 redis了解 缓存中间件 使用原因 缓解高并发、提升高可用。…

Qt 实现动态时钟

1.实现效果 2.widget.h #ifndef WIDGET_H #define WIDGET_H#include <QWidget>QT_BEGIN_NAMESPACE namespace

归一化输入

当输入的不同的特征取值范围差异过大&#xff0c;取得对应参数差别也会很大&#xff0c;在对参数进行优化的过程中&#xff0c;参数小的维度步长较小&#xff0c;参数大的维度步长较大&#xff0c;优化过程中路径曲折&#xff0c;将输入归一化&#xff0c;使特征取值范围差别小…

Leetcode 剑指 Offer II 098.不同路径

题目难度: 中等 原题链接 今天继续更新 Leetcode 的剑指 Offer&#xff08;专项突击版&#xff09;系列, 大家在公众号 算法精选 里回复 剑指offer2 就能看到该系列当前连载的所有文章了, 记得关注哦~ 题目描述 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下…

华强北耳机最强攻略。华强北Airpods不踩坑,指南在这

#华强北Airpods##华强北耳机#这一篇文章&#xff0c;我会比较啰嗦&#xff0c;但会十分详细介绍目前能入手的渠道 和每一个渠道入手的优缺点&#xff0c;以便各位选择适合自己的渠道入手。 ■ 01 芯片ic大升级—————— 采用全新07IC板的洛达Ae芯片 整体提升三个单位算法 该…

idea-java序列化serialversionUID自动生成

&#x1f496;简介 java.io.Serializable 是 Java 中的一个标记接口&#xff08;marker interface&#xff09;&#xff0c;它没有任何方法或字段。当一个类实现了 Serializable 接口&#xff0c;那么这个类的对象就可以被序列化和反序列化。序列化是将对象的状态转换为字节流…

【原创】java+ssm+mysql小区物业管理系统设计与实现

个人主页&#xff1a;程序猿小小杨 个人简介&#xff1a;从事开发多年&#xff0c;Java、Php、Python、前端开发均有涉猎 博客内容&#xff1a;Java项目实战、项目演示、技术分享 文末有作者名片&#xff0c;希望和大家一起共同进步&#xff0c;你只管努力&#xff0c;剩下的交…

使用 Docker-compose 部署达梦 DM 数据库

目录 1. 获取达梦 DM8 Docker 镜像并上传到 Harbor 服务器 2. Docker-compose 部署达梦 DM8 数据库 3. 配置 dm.ini 文件 4.完整的 dm.ini 文件 最近&#xff0c;将 MySQL 数据库迁移到了达梦 DM8 数据库。本文将分享如何通过 Docker-compose 部署达梦 DM8 数据库的过程&am…

全面的编程语言常识

本文首发 编程语言常识 语雀看图区别编程语言什么是强类型、弱类型语言&#xff1f;哪种更好&#xff1f;强...https://www.yuque.com/ysgstudyhard/da6e0c/ggatoo 看图区别编程语言 什么是强类型、弱类型语言&#xff1f;哪种更好&#xff1f; 强类型语言 强类型语言是一…

网络通信与并发编程(二)基于tcp的套接字、基于udp的套接字、粘包现象

基于tcp的套接字 文章目录 基于tcp的套接字一、套接字的工作流程二、基于tcp的套接字通信三、基于udp的套接字通信四、粘包现象 一、套接字的工作流程 Socket是应用层与TCP/IP协议族通信的中间软件抽象层&#xff0c;它是一组接口。在设计模式中&#xff0c;Socket其实就是一个…

【Java】多线程 Start() 与 run() (简洁实操)

Java系列文章目录 补充内容 Windows通过SSH连接Linux 第一章 Linux基本命令的学习与Linux历史 文章目录 Java系列文章目录一、前言二、学习内容&#xff1a;三、问题描述start() 方法run() 方法 四、解决方案&#xff1a;4.1 重复调用 .run()4.2 重复调用 start()4.3 正常调用…

基础数据结构——链表(单向链表,双向链表,循环链表)

1.概述 在计算机科学中&#xff0c;链表是数据元素的线性集合&#xff0c;其每个元素都指向下一个元素&#xff0c;元素存储上并不连续 分类 单向链表&#xff0c;每个元素只知道其下一个元素是谁 双向链表&#xff0c;每个元素知道其上一个元素和下一个元素 循环链表&am…

EasyExcel填充模板导出excel.xlsx

菜鸟的自我救赎&#xff0c;自从有了GPT&#xff0c;还是头一次一个bug写一天。 直接贴导出excel模板的完整案例 官网冲刺 EasyExcel EasyExcel填充模板导出excel.xlsx / 导出excel模板 一、bug(不需要请跳过) 1.1 使用apache poi操作excel报错 java.lang.NoSuchMethodError…