首页 > 数据库 >MySQL(十八)MySQL事务(二):事务的隔离级别

MySQL(十八)MySQL事务(二):事务的隔离级别

时间:2023-05-09 14:02:00浏览次数:55  
标签:事务 隔离 0.00 MySQL account sec mysql +----+----------+---------+

MySQL(十八)MySQL事务(二):事务的隔离级别


​ MySQL是一个客户端/服务器架构的软件,可以有若干个客户端与之连接,连接上之后都可以被称作是一个会话,每个客户端都可以在自己的会话中向服务器发出请求语句,一个请求语句可能是事务的一部分,因此对于服务器来说需要同时处理多个事务。由于事务具有隔离性,所以理论上事务对某个数据进行访问的时候,其他事务应该排队,等事务提交之后,其他事务才能继续执行,但是这样对性能影响太大。也就是需要在隔离性并发性之间做取舍。

1 数据准备

CREATE TABLE `student1` (
	`stu_no` INT,
	`name` VARCHAR(20),
	`class` VARCHAR(20),
	PRIMARY KEY(stu_no)
)ENGINE=INNODB CHARSET=utf8;

INSERT INTO student1 VALUES(1, 'A', '1班')

2 数据并发问题

​ 如果事务不能保证串行执行的情况下可能会出现的问题:

2.1 脏写(dirty write),也称作丢失更新

​ 如果一个事务修改了另一个未提交事务 修改过的数据,就发生了脏写。

image-20230503183946196

​ 如上脏写示意图,AB均开启事务,B首先将数据name='A'改为了李四,然后事务A又改成张三并提交刷盘,然后事务B进行回滚操作到自己事务的初始状态name='A',事务A的更新就丢失了,出现了脏写。

2.2 脏读

​ 一个事务读取了另一个事务更新完但还没有提交的数据被称作脏读。若后续事务回滚,则第一个事务读取到的内容就是临时且无效的。

image-20230503184358724
2.3 不可重复读

​ 一个事务读取了一个字段的同时,另一个事务对其进行了更新,当第一个事务再读的时候,发现字段值不同了。

image-20230503184926580
2.4 幻读

​ 一个事务在一张表中读取了一个字段之后,另一个事务插入了一些新的行,如果前一个事务再次读取同一个表,就会发现多了几行,这就被称作幻读。新插入的几行被称作幻影记录

image-20230503185254333
  • 如果删除了几行,导致读的记录变少了的情况算幻读吗

    不算,因为幻读强调是插入,即读到了之前没有读过的记录,删除的这几条严格来说是不可重复读

3 SQL的四种隔离级别

​ 上面介绍了并发执行事务可能出现的问题,这些问题并不需要全部解决,否则并发性能太差,因此可以按照业务场景解决问题较轻的,按严重程度排序:

image-20230503192031131
脏写 > 脏写 > 不可重复读 > 幻读

​ 可以牺牲一部分隔离性来换取事务的并发性,SQL标准中设置了四个隔离级别,隔离级别越低,事务的并发程度就越高:

  • 读未提交(READ UNCOMMITTED):在该隔离级别下,所有事务都能看到其他未提交事务的执行结果。不能避免脏读、不可重复读和幻读问题。
  • 读已提交(READ COMMITTED):在该隔离级别下,所有事务都只能看到其他已经提交事务的执行结果,可以避免脏读,但是不能解决不可重复读和幻读。这也是ORACLE数据库默认的隔离级别
  • 可重复读(REPEATABLE READ):在该隔离级别下,一个事务读取数据后,另一个事务对数据进行了修改提交,前一个事务读取的还是之前的结果,可以避免不可重复读和脏读,但是不能解决幻读。
  • 可串行化(SERIALIZABLE):确保事务串行执行,没有任何其他事务对表进行修改操作,所有的并发问题都可以解决,但效率低下。
image-20230503191751446

没有涉及脏写问题,是因为脏写问题太严重了:事务提交修改还能出错,因此上面的四种隔离模式都能避免脏写问题

4 MYSQL支持的隔离级别

​ 上面的四种隔离模式是SQL标准规定的,对于具体的数据库支持是不同的,比如ORACLE只支持读已提交可串行化,并且默认的隔离级别是读已提交,而MySQL支持四种隔离级别,而且默认的隔离级别是可重复度

4.1 MYSQL的隔离级别的设置与查看

MySQL的隔离级别的查看按照版本方式如下:

# 5.7之前
mysql> show variables like 'tx_isolation';

# 5.7之后
mysql> show variables like 'transaction_isolation';
+-----------------------+-----------------+
| Variable_name         | Value           |
+-----------------------+-----------------+
| transaction_isolation | REPEATABLE-READ |
+-----------------------+-----------------+
1 row in set (0.00 sec)
# 版本混用

mysql> select @@transaction_isolation;
+-------------------------+
| @@transaction_isolation |
+-------------------------+
| REPEATABLE-READ         |
+-------------------------+
1 row in set (0.00 sec)

可以按照下面的语句修改隔离级别:

SET [GLOBAL|SESSION] TRANSACTION ISOLATION LEVEL
> READ COMMITTED
> READ UNCOMMITTED
> REPEATABLE READ
> SERIALIZABLE

​ 或者:

SET [GLOBAL|SESSION] TRANSACTION_ISOLATION
> READ-COMMITTED
> READ-UNCOMMITTED
> REPEATABLE-READ
> SERIALIZABLE

关于设置GLOBAL或SESSION的影响:

  • 使用GLOBAL

    SET GLOBAL TRANSACTION ISOLATION READ-COMMITTED;
    
    • 当前会话所有后续事务有效(当前事务无效)
    • 对已经开启的事务执行期间运行,不会影响当前正在执行的事务
  • 使用SESSION

    SET GLOBAL TRANSACTION ISOLATION READ-COMMITTED;
    
    • 当前会话生效
    • 其他会话无效
4.2 不同隔离级别举例

数据准备

CREATE TABLE account (
	id INT PRIMARY KEY AUTO_INCREMENT,
	`name` VARCHAR(15),
	balance DECIMAL(10, 2)
);
INSERT INTO account(name, balance) 
VALUES('张三', 100), ('李四', 0);
不可重复读

​ 首先对两个会话都设置隔离模式为读未提交

mysql> set session transaction_isolation = 'read-uncommitted';
Query OK, 0 rows affected (0.00 sec)

mysql> select @@transaction_isolation;
+-------------------------+
| @@transaction_isolation |
+-------------------------+
| READ-UNCOMMITTED        |
+-------------------------+
1 row in set (0.00 sec)

会话一执行事务,但是没有提交:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> update account set balance = balance + 100 where id = 1;
Query OK, 1 row affected (0.00 sec)
Rows matched: 1  Changed: 1  Warnings: 0

会话二能够获取到未提交数据

mysql> select * from account;
+----+------+---------+
| id | name | balance |
+----+------+---------+
|  1 | ??   |  200.00 |
|  2 | ??   |    0.00 |
+----+------+---------+
2 rows in set (0.00 sec)

​ 此时会话一回滚,那么会话二获取的数据就是临时且无效的

mysql> rollback;
Query OK, 0 rows affected (0.00 sec)

​ 再举一个列子,如果会话一会话二转100,并没有提交事务:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)
mysql> update account set balance = balance - 100 where id = 1;
Query OK, 1 row affected (0.00 sec)
Rows matched: 1  Changed: 1  Warnings: 0

mysql> update account set balance = balance + 100 where id = 2;
Query OK, 1 row affected (0.00 sec)
Rows matched: 1  Changed: 1  Warnings: 0

mysql> select * from account;
+----+------+---------+
| id | name | balance |
+----+------+---------+
|  1 | ??   |    0.00 |
|  2 | ??   |  100.00 |
+----+------+---------+
2 rows in set (0.00 sec)

​ 此时事务二就读到了脏数据

mysql> select * from account;
+----+------+---------+
| id | name | balance |
+----+------+---------+
|  1 | ??   |    0.00 |
|  2 | ??   |  100.00 |
+----+------+---------+
2 rows in set (0.00 sec)

​ 而如果这时候事务二想把这些钱再转给事务一,就会发现卡住了,没有任何的执行结果

mysql> update account set balance = balance - 100 where id = 2;

​ 这是因为四种隔离级别都解决了脏写的问题,即事务二想修改未提交的事务是不能的,因为mysql对表进行了加锁,不允许其他事务对表进行修改

读已提交

数据准备

mysql> truncate table account;
Query OK, 0 rows affected (0.02 sec)

mysql> insert into account values(1, 'zhangsan', 100), (2, 'lisi', 0);
Query OK, 2 rows affected (0.00 sec)
Records: 2  Duplicates: 0  Warnings: 0

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |  100.00 |
|  2 | lisi     |    0.00 |
+----+----------+---------+
2 rows in set (0.00 sec)

​ 1、2均设置隔离级别为读已提交

set session transaction_isolation = 'read-committed';

​ 1账户金额减五十,但是事务不提交

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> update account set balance = balance - 50 where id = 1;
Query OK, 1 row affected (0.00 sec)
Rows matched: 1  Changed: 1  Warnings: 0

​ 2读取金额发现1的金额为100,没有读到脏数据,说明解决了脏读问题

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |  100.00 |
|  2 | lisi     |    0.00 |
+----+----------+---------+
2 rows in set (0.00 sec)

​ 1提交事务

commit;

​ 2再读仍然是50,说明隔离级别为读已提交不能够解决不可重复读问题。

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |  100.00 |
|  2 | lisi     |    0.00 |
+----+----------+---------+
2 rows in set (0.00 sec)

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |   50.00 |
|  2 | lisi     |    0.00 |
+----+----------+---------+
2 rows in set (0.00 sec)

可重复读

​ 设置1和2的隔离级别为可重复读

mysql> set transaction_isolation = 'repeatable-read';
Query OK, 0 rows affected (0.00 sec)

mysql> select @@transaction_isolation;
+-------------------------+
| @@transaction_isolation |
+-------------------------+
| REPEATABLE-READ         |
+-------------------------+
1 row in set (0.00 sec)

​ 2查看数据:

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |  100.00 |
|  2 | lisi     |    0.00 |
+----+----------+---------+
2 rows in set (0.00 sec)

​ 1将钱加50,并且不提交事务:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> update account set balance = balance + 50 where id = 1;
Query OK, 1 row affected (0.00 sec)
Rows matched: 1  Changed: 1  Warnings: 0

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |  100.00 |
|  2 | lisi     |    0.00 |
+----+----------+---------+
2 rows in set (0.00 sec)

​ 2查看数据,没有读到脏数据,说明可重复读解决了不可重复读问题

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |   50.00 |
|  2 | lisi     |    0.00 |
+----+----------+---------+
2 rows in set (0.00 sec)

​ 1提交事务

mysql> commit;
Query OK, 0 rows affected (0.00 sec)

​ 2再次查看,发现金额为100

可重复解决了一部分幻读问题

​ 1和2都是INNODB默认的隔离级别可重复读

mysql> select @@transaction_isolation;
+-------------------------+
| @@transaction_isolation |
+-------------------------+
| REPEATABLE-READ         |
+-------------------------+
1 row in set (0.00 sec)

​ 2查看表中的数据:

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |   50.00 |
|  2 | lisi     |    0.00 |
+----+----------+---------+
2 rows in set (0.00 sec)

​ 此时如果1添加记录再提交,并且没有提交事务

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |   50.00 |
|  2 | lisi     |    0.00 |
+----+----------+---------+
2 rows in set (0.00 sec)

mysql> insert into account(3, 'wangwu', 0);
ERROR 1064 (42000): You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '3, 'wangwu', 0)' at line 1

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> insert into account values(3, 'wangwu', 0);
Query OK, 1 row affected (0.00 sec)

​ 2再次查看表中数据:

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |   50.00 |
|  2 | lisi     |    0.00 |
+----+----------+---------+
2 rows in set (0.00 sec)

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |   50.00 |
|  2 | lisi     |    0.00 |
+----+----------+---------+
2 rows in set (0.00 sec)

这是不是说明可重复读解决了幻读问题呢?并不是,如果这时候2向表中添加一条数据,由于禁止脏写,所以如果1不提交事务的话,2是没有办法写数据的,所以1执行commit命令的同时,可以看到不允许插入主键为2的值,表名出现了幻读问题:

mysql> insert into account values(3, 'liliu', 1);
ERROR 1062 (23000): Duplicate entry '3' for key 'account.PRIMARY'

​ 2再次查看,这时候发现已经是三条数据了

mysql> select * from account;
+----+----------+---------+
| id | name     | balance |
+----+----------+---------+
|  1 | zhangsan |   50.00 |
|  2 | lisi     |    0.00 |
|  3 | wangwu   |    0.00 |
+----+----------+---------+
3 rows in set (0.00 sec)

可序列化

​ 在SERIALIZABLE隔离级别下,事务的执行会隐式地添加行(X)锁/gap(X)锁(也成间隙锁),事务一提交之后,事务二才能继续执行。


未完待续

标签:事务,隔离,0.00,MySQL,account,sec,mysql,+----+----------+---------+
From: https://www.cnblogs.com/tod4/p/17384656.html

相关文章

  • MySQL(十九)MySQL事务日志(一)RedoLog
    MySQL(十九)MySQL事务日志(一)RedoLog​ 事务的四种特性:原子性、一致性、持久性和隔离性都是基于什么机制实现的?事务的隔离性由锁机制实现而事务的原子性、一致性和持久性则由事务的redo和undo日志来实现的redolog是重做日志,提供再写入操作,恢复提交事务修改的页的操作......
  • mysql 索引
    mysql索引按存储方式区分: 一:b树,通常是使用b树这种方式,只有一个根节点,叶子节点之间彼此相连 二:hash,首先不能使用hash索引排序,并且它只支持等值索引,比如"=""in()""<=>"。 不支持键的部分匹配,因为计算hash值的时候是根据整体索引值来计算的。 建立hash索引相比b树需要更长的......
  • MySQL(十九)MySQL事务日志(二)UndoLog
    MySQL(十九)MySQL事务日志(二)UndoLog1undo日志概述​ redolog是事务持久性的保障,而undolog则是事务原子性和一致性的保证,如上图,在事务中更新数据的前置操作其实是需要将数据写入到undolog方便回滚。​ 事务需要保证原子性,也就是事务中的操作要么全部完成、要么全部不做。......
  • You have an error in your SQL syntax; check the manual that corresponds to your
    问题描述显示在条件查询的sql语句那里报错问题解决本来我是习惯了使用servlet写数据库操作的,然后就直接忽略掉了,或者说,直接忘记了在jsp里面的sql语句怎么正确书写了;经过查阅资料发现,查询语句是这样写的:Stringsql="select*frombookwhereid="+id;......
  • Centos7安装MySQL详细步骤(配置开机自启)
    MySQL检查系统是否安装过mysql//检查系统中有无安装过mysqlrpm-qa|grepmysql//查询所有mysql对应的文件夹,全部删除whereismysqlfind/-namemysql卸载CentOS7系统自带mariadb#查看系统自带的Mariadb[root@CDH-141~]#rpm-qa|grepmariadbmariadb-libs-5.5......
  • MYSQL查询【全部表】和表【所有字段】
    最近在做一些关于BI的东西。记录下。数据库查询全部表名称和备注信息只能查询表名SHOWTABLES; 查询表名和表备注SELECTTABLE_NAMEtablename,TABLE_COMMENTremarkFROMINFORMATION_SCHEMA.TABLESWHEREtable_schema="xxxx_xxxx"andT......
  • HTAP for MySQL 在腾讯云数据库的演进
    摘要:MySQL在充分利用多核计算资源方面比较欠缺,无法同时满足在线业务和分析型业务的客户需求,而单独部署一套专用的分析型数据库意味着额外的成本和复杂的数据链路。本次主题将介绍腾讯云数据库为满足此类场景而在HTAPforMySQL产品方面进行的尝试。2023首届云数据库技术沙龙MySQ......
  • mysql主从复制
    一、概述将主库的数据变更同步到从库,从而保证主库和从库数据一直。主要功能:数据备份、失败迁移、读写分离、降低单库读写压力二、原理1.主库会把数据变更记录在二进制日志文件Binlog中。2.从库连接主库,读取Binlog日志,并写入自身中继日志relaylog。3.slave重做中继日志,将......
  • mysql 8 安装
    1环境说明(1)基础环境操作系统:CentOSLinuxrelease7.4.1708(Core)配置:4c8GB100GBmysql版本:mysql-8.0.32-linux-glibc2.12-x86_64.tar.xz备份工具:percona-xtrabackup-8.0.32-26-Linux-x86_64.glibc2.17.tar.gz软件下载:wgethttps://downloads.percona.com/downloads/Per......
  • mysql 移动数据目录data后启动失败
    ###############################一、需要更改my.cnf中所有关于路径的参数二、更改binlog.index中的路径信息三、更改relaylog.index中的路径信息                   ###########################......