MySQL-事务

news/2024/11/18 0:46:47/

目录

🍁什么是事务

🍁隔离级别

🍃未提交读

🍃已提交读

🍃可重复读

🍃可串行化


🦐博客主页:大虾好吃吗的博客

🦐MySQL专栏:MySQL专栏地址

什么是事务

        多条sql语句,要么全部成功,要么全部失败。MySQL的事务是在存储引擎层实现。 MySQL的事务分别为ACID。

1. A 原子性(atomicity):一个事务必须被视为一个不可分割的单元。
2. C 一致性(consistency):数据库是从一种状态切换到另一种状态。
3. I 隔离性(isolation):事务在提交之前,对于其他事务不可见。
4. D 持久性(durablity):一旦事务提交,所修改的将永久保存到数据库。
mysql> create table bank(-> name varchar(24),-> money float);
Query OK, 0 rows affected (0.03 sec)
​
mysql> insert into bank values('z3',1000),('l4',5000);
Query OK, 2 rows affected (0.00 sec)
Records: 2  Duplicates: 0  Warnings: 0
Begin 或 start transaction开启事务
mysql> BEGIN;
Query OK, 0 rows affected (0.00 sec)
mysql> UPDATE bank SET MONEY=MONEY-1000 WHERE name='l4';
Query OK, 1 row affected (0.06 sec)
Rows matched: 1 Changed: 1 Warnings: 0
mysql> UPDATE bank SET MONEY=MONEY+1000 WHERE name='z3';
Query OK, 1 row affected (0.01 sec)
Rows matched: 1  Changed: 1  Warnings: 0
mysql> select * from bank;
+------+-------+
| name | money |
+------+-------+
| z3   |  2000 |
| l4   |  4000 |
+------+-------+
2 rows in set (0.01 sec)
mysql> rollback;
Query OK, 0 rows affected (0.00 sec)
​
mysql> select * from bank;
+------+-------+
| name | money |
+------+-------+
| z3   |  1000 |
| l4   |  5000 |
+------+-------+
2 rows in set (0.01 sec)
​
mysql> commit;
Query OK, 0 rows affected (0.00 sec)
​
mysql> select * from bank;
+------+-------+
| name | money |
+------+-------+
| z3   |  1000 |
| l4   |  5000 |
+------+-------+
2 rows in set (0.00 sec)

【总结事务命令】

事务开始: start transaction
事务开始: begin
事务提交: commit
回 滚: rollback

查看自动提交模式是自动还是手动

mysql> show variables like 'autocommit';
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| autocommit    | ON    |
+---------------+-------+
1 row in set (0.02 sec)
​
mysql> set autocommit=0;            #关闭自动提交
Query OK, 0 rows affected (0.00 sec)
​
mysql> show variables like 'autocommit';
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| autocommit    | OFF   |
+---------------+-------+
1 row in set (0.00 sec)

隔离级别


事务有4种隔离级别 事务在提交之前对其他事务可不可见

read unaommitted(未提交读)
read committed(已提交读)
Repeatable read(可重复读)
serializable(可串行化)

详细解释:

未提交读

事务中修改没有提交对其他事务也是可见的,俗称脏读

mysql> create table student(-> id int not null auto_increment,-> name varchar(30) not null default '',-> primary key (id)-> )engine=innodb auto_increment=2 default charset=utf8;Query OK, 0 rows affected (0.01 sec)

两端的客户端都设置成未提交读

mysql> set session tx_isolation='read-uncommitted';         #两台客户端都设置
Query OK, 0 rows affected, 1 warning (0.00 sec)

客户端A:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)
​
mysql> select * from student;
Empty set (0.00 sec)
​
mysql> insert into student(name) values('zhangsan');
Query OK, 1 row affected (0.00 sec)
mysql> //注意:此时事务未提交!!!

客户端B:

mysql> select * from student;
+----+----------+
| id | name     |
+----+----------+
|  2 | zhangsan |
+----+----------+
1 row in set (0.00 sec)

客户端B可以查看到信息

        总结:以上可以看出未提交读隔离级别非常危险,对于一个没有提交事务所做修改对另一个事务是可见状态,出现了脏读!非特殊情况不建议使用此级别。

已提交读

多数数据库系统默认为此级别(MySQL不是)。已提交读级别为一个事务只能已提交事务所做的修改,也就是解决了未提交读的问题

mysql> set session tx_isolation='read-committed';       #两台客户端都设置
Query OK, 0 rows affected, 1 warning (0.00 sec)

客户端A

mysql> begin;
Query OK, 0 rows affected (0.00 sec)
​
mysql> select * from student;
+----+----------+
| id | name     |
+----+----------+
|  2 | zhangsan |
+----+----------+
1 row in set (0.00 sec)
​
mysql> insert into student(name) values('lisi');
Query OK, 1 row affected (0.01 sec)

#此时去客户端B查看表,查看后在执行下面命令提交事务

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

客户端B:

mysql> select * from student;           #未提交事务前查看
+----+----------+
| id | name     |
+----+----------+
|  2 | zhangsan |
+----+----------+
1 row in set (0.00 sec) 
mysql> select * from student;           #提交事务后查看
+----+----------+
| id | name     |
+----+----------+
|  2 | zhangsan |
|  3 | lisi     |
+----+----------+
2 rows in set (0.00 sec)

        总结:从上面的例子可以看出,提交读没有了未提交读的问题,但是我们可以看到客户端A的一个事务中执行了两次同样的SELECT语句,,得到不同的结果,因此已提交读又被称为不可重复读。同样的筛选条件可能得到不同的结果。

可重复读

解决了不可重复读的问题,数据库级别没有解决幻读的问题。

mysql> set session tx_isolation='repeatable-read';          #两个客户端均设置为可重复读,然后两边一起开启一个事务
Query OK, 0 rows affected, 1 warning (0.00 sec)

客户端A:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)
​
mysql> select * from student;
+----+----------+
| id | name     |
+----+----------+
|  2 | zhangsan |
|  3 | lisi     |
+----+----------+
2 rows in set (0.00 sec)
​
mysql> update student set name='zhang3' where id=2;
Query OK, 1 row affected (0.01 sec)
Rows matched: 1  Changed: 1  Warnings: 0
​
mysql> commit;
Query OK, 0 rows affected (0.00 sec)

客户端B:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)
​
mysql> select * from student;
+----+----------+
| id | name     |
+----+----------+
|  2 | zhangsan |
|  3 | lisi     |
+----+----------+
2 rows in set (0.00 sec)
​
mysql> select * from student;
+----+----------+
| id | name     |
+----+----------+
|  2 | zhangsan |
|  3 | lisi     |
+----+----------+
2 rows in set (0.00 sec)
mysql> commit;
Query OK, 0 rows affected (0.00 sec)
​
mysql> select * from student;
+----+--------+
| id | name   |
+----+--------+
|  2 | zhang3 |
|  3 | lisi   |
+----+--------+
2 rows in set (0.00 sec)

        总结:上面的例子我们得知,可重复读两次读取的内容不一样。数据库的幻读问题并没有得到解决。幻读只读锁定里面的数据,不能读锁定外的数据,解决幻读出了mvcc机制Mvcc机制。

可串行化

是最高隔离级别,强制事务串行执行,执行串行了也就解决问题了,这个I别只有在对数据一致性要求非常严格并且没有并发的情况下使用

mysql> set session tx_isolation='serializable';             #两个客户端均设置为串读,然后两边一起开启一个事务
Query OK, 0 rows affected, 1 warning (0.00 sec)

客户端A:

mysql> begin;
Query OK, 0 rows affected (0.00 sec)
​
mysql> select * from student whereid < 10;
+----+--------+
| id | name   |
+----+--------+
|  2 | zhang3 |
|  3 | lisi   |
+----+--------+
2 rows in set (0.00 sec)

客户端B:

客户端B执行插入命令,发现卡顿,稍等一会就会返回ERROR。

mysql> insert into student(name) values('wangwu');
ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction

总结:我们发现INSERT 语句被阻塞执行,原因是A执行了查询表student同时满足id<10,已被锁定。如果查询表student同时满足id<5,则新增语句可以正常执行。

隔离级别

脏读

不可重复

幻读

加锁读

未提交读

提交读

可重复读

串行读


http://www.ppmy.cn/news/35141.html

相关文章

【Python入门第三十八天】Python丨NumPy 简介

什么是 NumPy&#xff1f; NumPy 是用于处理数组的 python 库。 它还拥有在线性代数、傅立叶变换和矩阵领域中工作的函数。 NumPy 由 Travis Oliphant 于 2005 年创建。它是一个开源项目&#xff0c;你可以自由使用它。 NumPy 指的是数值 Python&#xff08;Numerical Pyth…

SpringBoot @SpringBootTest 无法启动服务

这几天在看Hikari、Druid连接池。按照网上代码写Junit测试类。当时代码如下: package com.ceaning.crudp.utils;import org.junit.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; impo…

CAN(FD)记录仪在新能源汽车整车控制器(VCU)、电池管理系统(BMS)、电机控制器(MCU)、发动机ECU中的应用,免去出差烦恼

今天介绍CAN(FD)记录仪在新能源汽车整车控制器&#xff08;VCU&#xff09;、电池管理系统&#xff08;BMS&#xff09;、电机控制器&#xff08;MCU&#xff09;、发动机ECU中的应用 第一步&#xff1a;新能源汽车整车控制器&#xff08;VCU&#xff09;先供上电&#xff0c…

什么是分布式软件系统

:什么是分布式软件系统&#xff1f;分布式软件系统是什么意思&#xff1f; 分布式软件系统(Distributed Software Systems)是支持分布式处理的软件系统,是在由通信网络互联的多处理机体系结构上执行任务的系统。它包括分布式操作系统、分布式程序设计语言及其编译(解释)系统、分…

使用chatGPT实现数字自增动画

num-auto-add&#xff1a;数字自增动画 序言 我们经常在一些好的网站上遇到数字自增的动画效果&#xff0c;为用户提供了更加丰富的交互体验&#xff0c;看起来非常酷。 我之前也有写过&#xff0c;为了方便以后使用&#xff0c;打算将它优化&#xff0c;并上传到npm中。 首…

可别再用BeanUtils了(性能拉胯),试试这款转换神器

老铁们是不是经常为写一些实体转换的原始代码感到头疼&#xff0c;尤其是实体字段特别多的时候。有的人会说&#xff0c;我直接使用get/set方法。没错&#xff0c;get/set方法的确可以解决&#xff0c;而且也是性能较高的处理方法&#xff0c;但是大家有没有想过&#xff0c;要…

页面的重排和重绘?

思路&#xff1a; 网页渲染HTML文件到浏览器的过程->定义->如何优化网页渲染HTML文件到浏览器的过程HTML 文件通过HTML解析器解析生成DOM树&#xff1b;CSS文件通过CSS解析器生成CSSOM树&#xff1b;DOM树和CSSOM树生成渲染树&#xff08;render tree&#xff09;&#x…

Linux进程概念—环境变量

Linux进程概念—环境变量1.孤儿进程2.环境变量2.1常见环境变量2.2查看环境变量方法2.3在环境变量中添加2.4和环境变量相关的命令2.5环境变量的组织方式2.6命令行参数&#x1f31f;&#x1f31f;hello&#xff0c;各位读者大大们你们好呀&#x1f31f;&#x1f31f; &#x1f68…