MySql的事务问题

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: MySql的事务问题

MySQL事务及其特征

在正式讲解事务之前,我们先来说一下什么是事务。

事务(transaction)是用来维护数据库的完整性的,它可以保证一系列的MySQL操作 要么全部执行,要么全部不执行

我来举几个例子,来帮助大家理解,最经典的就是银行的转帐问题,比如说张三要转账给李四,我们是不是得保证张三账户的转出和李四账户的转入操作要么都成功,要么都失败,如果一个成功一个失败,就会导致转入金额和转出金额不一致的情况,为了防止这种情况,需要使用事务来处理。
在这里插入图片描述

事务的概念

接下来说一下事务的概念
事务(Transaction)指的是一个操作序列,该操作序列中的多个操作要么都做,要么都不做,是一个不可分割的工作单位,是数据库环境中的逻辑工作单位,由DBMS(数据库管理系统)中的事务管理子系统负责事务的处理。
目前常用的存储引擎有InnoDB(MySQL5.5以后默认的存储引擎)和MyISAM(MySQL5.5之前默认的存储引擎), 其中InnoDB支持事务处理机制,而MyISAM不支持。

事务可以保证多个操作原子性,要么全成功,要么全失败。对于数据库来说事务保证批量的DML要么全成功,要么全失败。事务具有四个特征ACID

事务的特性

在这里插入图片描述

a)原子性(Atomicity)
整个事务中的所有操作,必须作为一个单元全部完成(或全部取消)。
b)一致性(Consistency)
一致性是指事务执行的结果必须使数据库从一个一致性状态,变到另一个一致性状态。当数据库中只包含事务成功提交的结果时,数据库处于一致性状态。一致性是通过原子性来保证的。
例如:在转账时,只有保证转出和转入的金额一致才能构成事务。也就是说事务发生前和发生后,数据的总额依然匹配。
c)隔离性(Isolation): 一个事务不会影响其他事务的运行,多个事务并发要互相隔离。
也就是说:并发执行的事务之间既不能看到对方的中间状态,也不能相互影响。 就相当于有一堵墙隔在中间,这个墙可以很厚,也可以很薄,越厚隔离级别越高
例如:在转账时,只有当A账户中的转出和B账户中转入操作都执行成功后才能看到A账户中的金额减少以及B账户中的金额增多。并且其他的事务对于转账操作的事务是不能产生任何影响的。
d)持久性(Durability)
在事务完成以后,该事务对数据库所作的更改将持久地保存在数据库之中,并不会被回滚。
在正式讲解前,大家先来看一下一些概念
事务中存在一些概念:
a)事务(Transaction):一批操作(一组DML)
b)开启事务(Start Transaction)
c)回滚事务(rollback)
d)提交事务(commit)
e)SET AUTOCOMMIT:禁用或启用事务的自动提交模式

当执行DML语句是其实就是开启一个事务
关于事务的回滚需要注意:只能回滚insert、delete和update语句,不能回滚select(回滚select没有任何意义),对于create、drop、alter这些无法回滚.
事务只对DML有效果。
注意:rollback,或者commit后事务就结束了。

事务的演示

演示之前,给大家说一下 一些操作
start transaction--开启事务
savepoint 保存点名--设置保存点
rollback to 保存点名--回退事务
rollback--回退全部事务
commit-提交事务

先创建账户表

 create table account(
        id int primary key auto_increment,
        uname varchar(10) not null,
        balance double ); 

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
接下来,我就用下面的图来做一个小小的总结。
在这里插入图片描述
通过刚刚的演示,相信大家对事务已经有了一定的了解,接下来对其中的细节来进行说明

事务的细节说明

如果不开启事务,默认情况下,dml操作是自动提交的,不能回滚
在这里插入图片描述
MySQL的事务机制需要innodb的存储引擎才可以使用,myisam不好使。就是说InnoDB 存储引擎支持事务 , MyISAM 不支持
自动提交模式
•自动提交模式用于决定新事务如何及何时启动。
•启用自动提交模式:
–如果自动提交模式被启用,则单条DML语句将缺省地开始一个新的事务。
–如果该语句执行成功,事务将自动提交,并永久地保存该语句的执行结果。
–如果语句执行失败,事务将自动回滚,并取消该语句的结果。
–在自动提交模式下,仍可使用START TRANSACTION语句来显式地启动事务。这时,一个事务仍可包含多条语句,直到这些语句被统一提交或回滚。
•禁用自动提交模式:
–如果禁用自动提交,事务可以跨越多条语句。
–在这种情况下,事务可以用COMMIT和ROLLBACK语句来显式地提交或回滚。
•自动提交模式可以通过服务器变量AUTOCOMMIT来控制。
•例如:
mysql> SET AUTOCOMMIT = OFF;
mysql> SET AUTOCOMMIT = ON;

mysql> SET SESSION AUTOCOMMIT = OFF;
mysql> SET SESSION AUTOCOMMIT = ON;
show variables like '%auto%'; -- 查看变量状态
在这里插入图片描述

事务的并发问题

脏读(Dirty read)

当一个事务正在访问数据并且对数据进行了修改,而这种修改还没有提交到数据库中,这时另外一个事务也访问了这个数据,然后使用了这个数据。因为这个数据是还没有提交的数据,那么另外一个事务读到的这个数据是“脏数据”,依据“脏数据”所做的操作可能是不正确的。
简单来说就是一个事务读取到了另外一个事务还没有提交的数据,而且这个数据可能是错误的,后面可能会回滚。
在这里插入图片描述

不可重复读

(Unrepeatableread): 指在一个事务内多次读同一数据。在这个事务还没有结束时,另一个事务也访问该数据。那么,在第一个事务中的两次读数据之间,由于第二个事务的修改导致第一个事务两次读取的数据可能不太一样。这就发生了在一个事务内两次读到的数据是不一样的情况,因此称为不可重复读。
简单来说就是一个事务两次读取的数据不一样

在这里插入图片描述

幻读

(Phantom read): 幻读与不可重复读类似。它发生在一个事务(T1)读取了几行数据,接着另一个并发事务(T2)插入了一些数据时。在随后的查询中,第一个事务(T1)就会发现多了一些原本不存在的记录,就好像发生了幻觉一样,所以称为幻读。

在这里插入图片描述

不可重复度和幻读区别:

不可重复读的重点是修改,幻读的重点在于新增或者删除。

解决不可重复读的问题只需锁住满足条件的行,解决幻读需要锁表 ,因为我们不知道新增或者删除的是哪一行记录,可能新增或删除的是我们锁住的行的其他行

例1(同样的条件, 你读取过的数据, 再次读取出来发现值不一样了 ):事务1中的A先生读取自己的工资为 1000的操作还没完成,事务2中的B先生就修改了A的工资为2000,导 致A再读自己的工资时工资变为 2000;这就是不可重复读。

例2(同样的条件, 第1次和第2次读出来的记录数不一样 ):假某工资单表中工资大于3000的有4人,事务1读取了所有工资大于3000的人,共查到4条记录,这时事务2 又插入了一条工资大于3000的记录,事务1再次读取时查到的记录就变为了5条,这样就导致了幻读

事务的隔离级别

1.详细介绍+演示

我们知道同一时间,MySQL可能有多个连接,当多个连接开启各自的事务操作来操作数据库中的数据的时候,数据库系统复制隔离操作,用来保证各个连接在获取数据时候的准确性。
如果不考虑隔离性,就可能会引发脏读,不可重复读,幻读。
MySQL中提供了四种隔离级别来解决上述问题。
事务的隔离级别从低到高依次为READ UNCOMMITTED、READ COMMITTED、REPEATABLE READ以及SERIALIZABLE,隔离级别越低,越能支持高并发的数据库操作。

InnoDB 实现了四个隔离级别,用以控制事务所做的修改,并将修改通告至其它并发的事务:
–读未提交(READ UMCOMMITTED 最低的隔离级别)

允许一个事务可以看到其他事务未提交的修改。比如有两间教室,中间有墙隔开,但是墙很薄,一间教室的人说话,另外一间教室的人也可以听到

–读已提交(READ COMMITTED)

允许一个事务只能看到其他事务已经提交的修改,未提交的修改是不可见的。

–可重复读(REPEATABLE READ) --MySQL默认隔离级别

确保如果在一个事务中执行两次相同的SELECT语句,都能得到相同的结果,不管其他事务是否提交这些修改。 (银行总账)简单来说就是事务A开启以后,每一次读到的数据都是一样的,就算是事务B修改数据,事务A读取的数据也不会改变
该隔离级别为InnoDB的缺省设置。      
–串行化(SERIALIZABLE) 【序列化】
将一个事务与其他事务完全地隔离。 

例:A可以开启事务,B也可以开启事务
A在事务中执行DML语句时,未提交
B不可以执行DML,DQL语句
也就是说事务要排队执行,不可以并发操作。
在这里插入图片描述

虽然说了一大堆的理论性的东西,但是如果没有实际操作的话,其实是很难懂的,接下来我就进行演示,大家也要自己动手试试。
隔离级别演示
读未提交(其他的类似,大家自己操作)在这里插入图片描述
1.用两个控制台来演示
2.查看当前事务的隔离级别

在这里插入图片描述
3.设置当前会话隔离级别
set session transaction isolation read uncommitted;
在这里插入图片描述
注意:隔离级别是和事务相关的,离开事务就不要谈隔离级别了

4.开启事务
我们先创建表

可以看出现在表中没有数据

mysql> create table account(
    -> id int,
    -> `name` varchar(32),
    -> money int);
Query OK, 0 rows affected (0.04 sec)

mysql> select * from account;
Empty set (0.01 sec)

下面那个现象就是脏读
在这里插入图片描述
脏读和幻读
在这里插入图片描述
在这里插入图片描述

2.设置隔离级别

在这里插入图片描述

通过修改配置文件设置
可以在my.ini文件中使用transaction-isolation选项来设置服务器的缺省事务隔离级别。
该选项值可以是:
–READ-UNCOMMITTED
–READ-COMMITTED
–REPEATABLE-READ
–SERIALIZABLE

•例如:
[mysqld]
transaction-isolation = READ-COMMITTED
在这里插入图片描述
保存以后,要重新起点MySQL服务才会生效
通过命令动态设置隔离级别
•隔离级别也可以在运行的服务器中动态设置,应使用SET TRANSACTION ISOLATION LEVEL语句。
•其语法模式为:

SET [GLOBAL | SESSION] TRANSACTION ISOLATION LEVEL <isolation-level>
其中的<isolation-level>可以是:

–READ UNCOMMITTED
–READ COMMITTED
–REPEATABLE READ
–SERIALIZABLE
•例如: SET TRANSACTION ISOLATION LEVEL REPEATABLE READ;

3.隔离级别的作用范围

•事务隔离级别的作用范围分为两种:
–全局级:对所有的会话有效
–会话级:只对当前的会话有效
•例如,设置会话级隔离级别为READ COMMITTED :
mysql> SET TRANSACTION ISOLATION LEVEL READ COMMITTED;
或:
mysql> SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;
•设置全局级隔离级别为READ COMMITTED :
mysql> SET GLOBAL TRANSACTION ISOLATION LEVEL READ COMMITTED;

4、查看隔离级别

•服务器变量tx_isolation(包括会话级和全局级两个变量)中保存着当前的会话隔离级别。
•为了查看当前隔离级别,可访问tx_isolation变量:
–查看会话级的当前隔离级别:

mysql> SELECT @@tx_isolation;
或: 
mysql> SELECT @@session.tx_isolation;

–查看全局级的当前隔离级别:

mysql> SELECT @@global.tx_isolation;

相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
7天前
|
SQL 安全 关系型数据库
【MySQL基础篇】事务(事务操作、事务四大特性、并发事务问题、事务隔离级别)
事务是MySQL中一组不可分割的操作集合,确保所有操作要么全部成功,要么全部失败。本文利用SQL演示并总结了事务操作、事务四大特性、并发事务问题、事务隔离级别。
【MySQL基础篇】事务(事务操作、事务四大特性、并发事务问题、事务隔离级别)
|
13天前
|
SQL 关系型数据库 MySQL
MySQL进阶突击系列(04)事务隔离级别、AICD、CAP、BASE原则一直搞不懂? | 看这篇就够了
本文详细介绍了数据库事务的四大特性(AICD原则),包括原子性、隔离性、一致性和持久性,并深入探讨了事务并发问题与隔离级别。同时,文章还讲解了分布式系统中的CAP理论及其不可能三角关系,以及BASE原则在分布式系统设计中的应用。通过具体案例和图解,帮助读者理解事务处理的核心概念和最佳实践,为应对相关技术面试提供了全面的知识准备。
|
3月前
|
存储 SQL 关系型数据库
MySQL的事务隔离级别
【10月更文挑战第17天】MySQL的事务隔离级别
133 43
|
2月前
|
关系型数据库 MySQL
mysql事务特性
原子性:一个事务内的操作统一成功或失败 一致性:事务前后的数据总量不变 隔离性:事务与事务之间相互不影响 持久性:事务一旦提交发生的改变不可逆
|
2月前
|
关系型数据库 MySQL 数据库
MySQL事务隔离级别及默认隔离级别的设置
在数据库系统中,事务隔离级别是一个关键的概念,它决定了事务在并发执行时如何相互隔离。MySQL提供了四种事务隔离级别,每种级别都解决了不同的并发问题。本文将详细介绍这些隔离级别以及MySQL的默认隔离级别。
|
3月前
|
存储 缓存 关系型数据库
MySQL事务日志-Redo Log工作原理分析
事务的隔离性和原子性分别通过锁和事务日志实现,而持久性则依赖于事务日志中的`Redo Log`。在MySQL中,`Redo Log`确保已提交事务的数据能持久保存,即使系统崩溃也能通过重做日志恢复数据。其工作原理是记录数据在内存中的更改,待事务提交时写入磁盘。此外,`Redo Log`采用简单的物理日志格式和高效的顺序IO,确保快速提交。通过不同的落盘策略,可在性能和安全性之间做出权衡。
1721 14
|
4月前
|
存储 Oracle 关系型数据库
Oracle和MySQL有哪些区别?从基本特性、技术选型、字段类型、事务、语句等角度详细对比Oracle和MySQL
从基本特性、技术选型、字段类型、事务提交方式、SQL语句、分页方法等方面对比Oracle和MySQL的区别。
758 18
|
3月前
|
SQL 关系型数据库 MySQL
阿里面试:MYSQL 事务ACID,底层原理是什么? 具体是如何实现的?
尼恩,一位40岁的资深架构师,通过其丰富的经验和深厚的技術功底,为众多读者提供了宝贵的面试指导和技术分享。在他的读者交流群中,许多小伙伴获得了来自一线互联网企业的面试机会,并成功应对了诸如事务ACID特性实现、MVCC等相关面试题。尼恩特别整理了这些常见面试题的系统化解答,形成了《MVCC 学习圣经:一次穿透MYSQL MVCC》PDF文档,旨在帮助大家在面试中展示出扎实的技术功底,提高面试成功率。此外,他还编写了《尼恩Java面试宝典》等资料,涵盖了大量面试题和答案,帮助读者全面提升技术面试的表现。这些资料不仅内容详实,而且持续更新,是求职者备战技术面试的宝贵资源。
阿里面试:MYSQL 事务ACID,底层原理是什么? 具体是如何实现的?
|
4月前
|
SQL 关系型数据库 MySQL
MySQL基础:事务
本文详细介绍了数据库事务的概念及操作,包括事务的定义、开启、提交与回滚。事务作为一组不可分割的操作集合,确保了数据的一致性和完整性。文章还探讨了事务的四大特性(原子性、一致性、隔离性、持久性),并分析了并发事务可能引发的问题及其解决方案,如脏读、不可重复读和幻读。最后,详细讲解了不同事务隔离级别的特点和应用场景。
185 4
MySQL基础:事务
|
3月前
|
SQL 关系型数据库 MySQL
【MySQL】索引和事务
【MySQL】索引和事务
63 0