mysql--事务四大特性与隔离级别

事务四大特性与隔离级别

mysql事务的概念

事务就是一组操作的集合,他是一个不可分割的整体,事务会把所有的操作作为一个整体一起向系统提交或撤销操作。

事务的属性

  • 原子性(Atomicity)
  • 一致性(Consistency)
  • 隔离性(Isolation)
  • 持久性(Durability)

事务控制语句

  • 查看/设置事务提交
sql 复制代码
START TRANSACTION;
  • 提交一个事务
sql 复制代码
COMMIT;
  • 回滚一个事务
sql 复制代码
ROLLBACK;

1为自动提交,0位手动提交。

msyql默认为自动提交,需要手动设置为手动提交。

执行成功时,进行commit提交事务,程序出时是进行回滚事务

转账示例

sql 复制代码
-- 关闭自动提交
SET AUTOCOMMIT = 0;

-- 开始事务
START TRANSACTION;

-- 查询张三和李四的余额
SELECT balance INTO @zhangsan_balance FROM accounts WHERE name = '张三';
SELECT balance INTO @lisi_balance FROM accounts WHERE name = '李四';

-- 如果张三的余额足够,进行转账操作
IF @zhangsan_balance >= 1000 THEN
    -- 更新张三和李四的余额
    UPDATE accounts SET balance = balance - 1000 WHERE name = '张三';
    UPDATE accounts SET balance = balance + 1000 WHERE name = '李四';
    -- 提交事务
    COMMIT;
    SELECT '转账成功';
ELSE
    -- 回滚事务
    ROLLBACK;
    SELECT '转账失败,张三余额不足';
END IF;

-- 开启自动提交
SET AUTOCOMMIT = 1;

并发事务引发的问题

脏读

脏读(Dirty Read)是指一个事务在读取另一个未提交的事务所做的修改时,产生不一致的结果。具体来说,当一个事务读取了另一个事务未提交的数据,然后这个未提交的数据被回滚了,那么读取到的数据就是脏数据,因为它实际上不应该存在于数据库中。

脏读场景

假设有两个用户:A 和 B。用户 A 有 1000 元,用户 B 有 2000 元。现在 A 从自己的账户中转账 500 元到 B 的账户。

  1. 初始状态

用户 A 的账户余额为 1000 元。

用户 B 的账户余额为 2000 元。

  1. 转账操作

用户 A 开始一个事务,从自己的账户中扣除 500 元。

用户 B 开始一个事务,将 500 元存入自己的账户。

  1. 脏读

用户 B 在事务中读取自己的账户余额,发现账户余额变成了 2500 元(未提交的转账操作)。

用户 A 的转账操作因为某种原因失败,并回滚了。

  1. 结果

用户 B 实际上没有收到任何转账,但是由于脏读,他错误地读取到了自己账户的余额为 2500 元,这是一个不一致的状态。

不可重复读

事务A在第一步查询了id为1的数据,然后事务B修改id为1的数据并且提交了数据,然后事务A在第三步又查询了id为1的数据,发现两次查询的数据不一致。

幻读

幻读(Phantom Read)是并发事务可能遇到的另一种问题,也涉及到事务隔离性。幻读场景指的是在一个事务中多次执行相同的查询,但在不同的查询中返回了不同数量的行。

幻读场景

  • 事务T1执行了一个查询,返回了一组满足某个条件的行。
  • 事务T2插入了一些新的行,这些行也满足了T1查询中的条件,并且提交了事务。
  • 事务T1再次执行相同的查询,但这次返回的结果集比之前多了新插入的行,导致得到了不一致的结果。

事务的隔离级别

对号表示问题会出现,X白表示问题已解决。

串行化·:安全性高,效率低。

读未提交:效率最高,但是安全性最低

读未提交

sql 复制代码
-- 事务T1
START TRANSACTION;

-- 事务T2
START TRANSACTION;

-- T2更新了某行数据
UPDATE table_name SET column1 = new_value WHERE condition;

-- T1读取T2尚未提交的数据
SELECT * FROM table_name WHERE condition;

-- T2提交
COMMIT;

-- T1继续
-- T1读取了已提交的数据
SELECT * FROM table_name WHERE condition;

-- T1提交
COMMIT;

读已提交

sql 复制代码
-- 事务T1
START TRANSACTION;

-- 事务T2
START TRANSACTION;

-- T2更新了某行数据
UPDATE table_name SET column1 = new_value WHERE condition;

-- T2提交
COMMIT;

-- T1读取T2已提交的数据
SELECT * FROM table_name WHERE condition;

-- T1提交
COMMIT;

可重复读(MySQL默认)

sql 复制代码
-- 事务T1
START TRANSACTION;

-- 事务T2
START TRANSACTION;

-- T2更新了某行数据
UPDATE table_name SET column1 = new_value WHERE condition;

-- T2提交
COMMIT;

-- T1再次读取T2修改的数据,但是在可重复读级别下,T1看到的数据仍然是之前启动事务时的快照数据
SELECT * FROM table_name WHERE condition;

-- T1提交
COMMIT;

总结

相关推荐
松涛和鸣4 小时前
72、IMX6ULL驱动实战:设备树(DTS/DTB)+ GPIO子系统+Platform总线
linux·服务器·arm开发·数据库·单片机
likangbinlxa4 小时前
【Oracle11g SQL详解】UPDATE 和 DELETE 操作的正确使用
数据库·sql
r i c k5 小时前
数据库系统学习笔记
数据库·笔记·学习
野犬寒鸦5 小时前
从零起步学习JVM || 第一章:类加载器与双亲委派机制模型详解
java·jvm·数据库·后端·学习
IvorySQL6 小时前
PostgreSQL 分区表的 ALTER TABLE 语句执行机制解析
数据库·postgresql·开源
·云扬·6 小时前
MySQL 8.0 Redo Log 归档与禁用实战指南
android·数据库·mysql
IT邦德6 小时前
Oracle 26ai DataGuard 搭建(RAC到单机)
数据库·oracle
惊讶的猫6 小时前
redis分片集群
数据库·redis·缓存·分片集群·海量数据存储·高并发写
不爱缺氧i7 小时前
完全卸载MariaDB
数据库·mariadb
纤纡.7 小时前
Linux中SQL 从基础到进阶:五大分类详解与表结构操作(ALTER/DROP)全攻略
linux·数据库·sql