mysql事务的理解学习, 面试不问索引原理就是事务原理

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
云数据库 RDS MySQL,高可用系列 2核4GB
简介: mysql事务的理解学习, 面试不问索引原理就是事务原理

事务理解

事务组成

  • 简单的来说事务可以由一条简单的sql语句组成,也可以由一组复杂的sql语句组成(事务是一个程序逻辑单元)

事务特征

在数据库针对事务进行提交的时候,要么是所有的修改都保存,要么所有的修改都丢弃  (原子性,要么整个事务全部完成,要么整个事务全部丢弃)


事务是访问并更新数据库各种数据项的一个程序执行单元


mysql的innodb 引擎是支持事务的, myisam 是不支持事务的, innodb中每一条sql语句都是事务;我们如果想要取消自动提交事务,可以通过设置 set autocommit = 0; 设置当前会话手动提交事务


常见事务的控制语句

-- 显示的开启事务
start transaction | begin;
-- 提交事务,并且对于当前数据库所作的操作修改做持久化
commit;
-- 回滚事务,结束用户的事务,并撤销正在进行的所有未提交的修改
rollback;
-- 创建一个保存点,一个事务可以有多个保存点
savepoint identifier
-- 删除一个保存点
release savepoint identifier
-- 事务回滚到保存点
rollback to [savepoint] identifier

事务的ACID特征

原子性 (可借助多线程对全局数据做写入操作哪里的理解)  绑定操作单元


浅层理解:这个好理解, 就是说事务一定是一个原子操作序列单元,所有操作在一次执行过程中仅仅只会允许两种状态                ---   要么全部执行成功  要么全部执行失败   绑定在一个事务中的所有操作


深入理解:事务操作要么都做(提交),要么都不做(回滚);事务是访问并更新数据库各种数据项的一个程 序执行单元,是不可分割的工作单位;通过undolog来实现回滚操作。undolog记录的是事务每步具体操作,当回滚时,回放事务具体操作的逆运算;(回滚原理,沿着undolog往回恢复操作之前的样子)


隔离性


一个事务的执行不能被其他事务干扰。


要解释隔离性: 首先理解事务并发,我们的数据库的用户是很多的,是可以并发执行事务的,隔离性指的就是并发的事务是相互隔离,互不干扰的。 也就是说不同的事务并发操作相同的数据时候,每一个事务都是存在自己独立完整的数据空间的


深入理解原理:事务的隔离性要求每个读写事务的对象对其他事务的操作对象能相互分离,也就是事务提交前对其他事务都不可见;通过 MVCC 和 锁来实现;MVCC 时多版本并发控制,主要解决一致性非锁定读,通过记录和获取行版本,而不是使用锁来限制读操作,从而实现高效并发读性能。锁用来处理 并发 DML 操作;数据库中提供粒度锁的策略,针对表(聚集索引B+树)、页(聚集索引B+树叶 子节点)、行(叶子节点当中某一段记录行)三种粒度加锁

MVCC性能更优, 避免了加行锁读操作,避免了锁等待


持久性  (先写日志再写磁盘, 高效)


事务一旦提交,他对数据的改动是持久性的,事务一旦提交,相关的数据就应该从游离态或瞬时态变成持久态。


redo log是InnoDB引擎特有的日志模块,记录的是:在某个数据页上做了什么修改;


当MySQL执行一条更新语句的时候,InnoDB引擎会把记录先写到redo log文件中,并更新内存。此时这条更新操作就算完成了,但是并没有将数据的更新持久化的磁盘,InnoDB引擎会在一个合适的时机将数据的更新操作持久化到磁盘。


事务提交后,事务DML操作将会持久化(写入redolog磁盘文件 哪一个页 页偏移值 具体数据); 即使发生宕机等故障,数据库也能将数据恢复。redo log记录的是物理日志; (意思就是说事务的所有的DML操作都会写入redo log 磁盘文件, 将操作的位置和具体操作都记录在redo log日志中, 所以就算是发生了宕机等现象也是可以还原的)


一致性  (是一种状态, 无论是回滚还是提交, 数据完整性不被破坏)


指的是数据库操作前后需要保持数据的完整性和一致性,  事务的执行是导致数据库从一种一致性状态转换为另外一种一致性状态,也就是说事务执行前后,数据库的完整性约束没有被破坏.       (eg : A账户向B账户转账500元,不可能说A账户减少500元,B账户却没有增加500元)


一致性由原子性、隔离性以 及持久性共同来维护的。

事务的并发异常

脏读

  • 事务A可以读取到另外一个事务B还没有提交的数据这个事务A读取到的未提交数据就是所谓的脏读数据,要演示读取到脏读数据, 首先我们需要将隔离等级设置为 read uncommitted 读未提交
  • 演示脏读

不可重复读  (读 + 读,结果不一样)


事务(A) 可以读到另外一个事务(B)中提交的数据;通常发生在一个事务中两次读到的数据是不一样的情况; 不可重复读在隔离级别 READ COMMITTED 存在。一般而言,不可重复读的问题是可以接受的,因为读到已经提交的数据,一般不会带来很大的问题,所以很多厂商(如Oracle、 SQL Server)默认隔离级别就是READ COMMITTED


A 事务 第一次读 , B事务还没有commit , A 事务第二次读, B事务进行了commit, 两次读结果就不会一致

幻读        (读 + 写, 以为自己读到了,去对其进行写操作,实际不存在)


事务中一次读操作不能支撑接下来的业务逻辑;通常发生在一个事务中一次读判断接下来写操作失败的情况;   例如:以name为唯一键的表,一个事务中查询 select * from t where name = 'mark'; 不存在,接下来 insert into t(name) values ('mark'); 出现错误,此时另外一个 事务也执行了 insert 操作;幻读在隔离级别 repeatable read 及以下存在;但是可以在 repeatable read 级别下通过读加锁(使用共享锁)解决;


简单的举出一个幻读的例子,就是A事务刚刚查询一个记录不存在,接下来准备做insert操作,这个时候另外的B事务插入了这个记录,并提交,此时相当于是刚刚的记录查询是虚幻的,不是真实的,也称为幻读  

解决办法1:  查询操作的时候加上一把共享读锁, 读锁,间隙锁,锁住之后别的事务就不能对齐进行插入操作了, 查询的时候加上共享锁,事务B 就无法完成插入操作, 这样就避免了幻读现象了   (mysql中的共享锁就相当于我们学操作系统时候的写锁)


解决办法2: 直接修改事务的隔离级别为顺序读: serializable,  串行之后会读操作也会自动加上共享锁 share in mode,但是基本不提倡,效率太低下了,锁等待,性能下降的可怕.

事务隔离级别

  • 隔离级别的浅显知道

事务隔离等级越高,会越来越安全, 但是效率会变得越来越低,并发性也越来越差


默认等级一般是读已提交或者是可重复读


ISO和ANIS SQL标准制定了四种事务隔离级别的标准,各数据库厂商在正确性和性能之间做了妥 协,并没有严格遵循这些标准;MySQL innodb默认支持的隔离级别是 REPEATABLE READ;    


隔离级别的底层实现 (mvcc  + 各种锁)    mvcc + 各种锁都是为了保证在并发执行下事务之间的隔离性, 原子性 等性质

READ UNCOMMITTED 读未提交;该级别下读不加锁,写加排他锁,写锁在事务提交或回滚后释放锁       (读未提交,这个级别下读是不会加锁的,也没有mvcc)


READ COMMITTED  读已提交;  该级别下读不会自动加锁, 而是采取的mvcc(多版本并发控制),  也就是提供一致性非锁定读,  此时读取操作 读取历史快照数据;该隔离级别下读取历史版本的最新数据,所以读取的是已提交的数据;      (正是因为读的是历史版本最新的数据, 也就是当前已提交的数据, 所以不支持重复读,因为有其他事务做新的提交, 该事务重复再读取结果会和提交前不一致)

REPEATABLE READ 可重复读;该级别下也支持 MVCC,此时读取操作读取事务开始时的版本数据


也就是说 repeatable read 如何支持可重复读, 不管你提交多少新的版本, 我永远只是读取开启事务的第一版, 也就是我最初的一版历史数据, 其他事务提交的最新版历史快照数据我不会理睬.      (      因为一直读取第一版数据, 自然不会因为最新版数据引起重复读数据不一致的问题, 这个就是 reapeatable read, 所以现在懂了塞,为啥它可以支持重复读数据一致     )


SERIALIZABLE 可串行化;该级别下给读加了共享锁 (S锁);所以事务都是串行化的执行;此时隔离级别最严苛;

小总结

首先本章主要学习事务的定义,了解事务的重要性质, 初步认识锁 + mvcc 支持并发多事务执行操做的原因  (以及认识各种隔离级别的底层实现)


啥叫事务:一条简单的sql语句 或者 是 多条复杂的sql语句的集合


事务需要保证哪些性质        :  


原子性: undo log  回滚操作保证所有的sql语句要么全部执行成功,要么全部执行失败


隔离性:   mvcc + 锁 保证 读写操作 各个事务之间相互隔离,互不干扰,  写操作肯定要加x锁, 读操作  分版本,


read uncommitted 不会加锁也没有mvcc  


read committed 存在mvcc多版本并发控制,   可能出现幻读现象,可以s共享锁避免, 读最新版本数据,存在不可重复读问题,  

             

repeatable read  读取最老版本的数据 避免重复读数据不一致的问题, 但是也可能出现幻读, 解决办法还是加s共享锁


serializable: 顺序读,读操作自动s锁, 绝对安全,但是并发性是最差的    

持久化:使用redo log 支持持久化, 对于mysql数据的修改物理地址 +  如何修改做日志记录,  就算是宕机了也可以利用 redo log 日志进行恢复


一致性 : 保证事务前后数据的完整性,从一种一致性状态转换为另外一种一致性状态, 一致性的保证依赖的是上述三种性质


read committed 读最新版本数据会出现 不可重复读的问题  (读取的数据是其他事务提交前和提交后的不同结果)


repeatable read : 可重复读,读历史快照数据的第一版数据,避免重复读结果不一致,但是可能出现幻读的现象,幻读,读 + 写 读完根据读结果进行 update 操作。问题,刚读完,其他事务进行了update + 提交。此刻之前select的结果已然是不正确了,之前读的数据就是幻读数据了   (如何避免,加上 s锁)


留个后文,mysql各种锁的详解,锁----》 事务至关重要


相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助     相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
17天前
|
SQL 存储 关系型数据库
MySQL秘籍之索引与查询优化实战指南
最左前缀原则。不冗余原则。最大选择性原则。所谓前缀索引,说白了就是对文本的前几个字符建立索引(具体是几个字符在建立索引时去指定),比如以产品名称的前 10 位来建索引,这样建立起来的索引更小,查询效率更快!
81 22
 MySQL秘籍之索引与查询优化实战指南
|
17天前
|
SQL 关系型数据库 MySQL
MySQL事务日志-Undo Log工作原理分析
事务的持久性是交由Redo Log来保证,原子性则是交由Undo Log来保证。如果事务中的SQL执行到一半出现错误,需要把前面已经执行过的SQL撤销以达到原子性的目的,这个过程也叫做"回滚",所以Undo Log也叫回滚日志。
MySQL事务日志-Undo Log工作原理分析
|
6天前
|
Java Linux 调度
硬核揭秘:线程与进程的底层原理,面试高分必备!
嘿,大家好!我是小米,29岁的技术爱好者。今天来聊聊线程和进程的区别。进程是操作系统中运行的程序实例,有独立内存空间;线程是进程内的最小执行单元,共享内存。创建进程开销大但更安全,线程轻量高效但易引发数据竞争。面试时可强调:进程是资源分配单位,线程是CPU调度单位。根据不同场景选择合适的并发模型,如高并发用线程池。希望这篇文章能帮你更好地理解并回答面试中的相关问题,祝你早日拿下心仪的offer!
25 6
|
18天前
|
存储 关系型数据库 MySQL
MySQL中为什么要使用索引合并(Index Merge)?
通过这些内容的详细介绍和实际案例分析,希望能帮助您深入理解索引合并及其在MySQL中的
69 10
|
26天前
|
存储 关系型数据库 MySQL
【MYSQL】 ——索引(B树B+树)、设计栈
索引的特点,使用场景,操作,底层结构,B树B+树,MYSQL设计栈
|
1天前
|
缓存 关系型数据库 MySQL
【深入了解MySQL】优化查询性能与数据库设计的深度总结
本文详细介绍了MySQL查询优化和数据库设计技巧,涵盖基础优化、高级技巧及性能监控。
13 0
|
28天前
|
存储 Oracle 关系型数据库
数据库传奇:MySQL创世之父的两千金My、Maria
《数据库传奇:MySQL创世之父的两千金My、Maria》介绍了MySQL的发展历程及其分支MariaDB。MySQL由Michael Widenius等人于1994年创建,现归Oracle所有,广泛应用于阿里巴巴、腾讯等企业。2009年,Widenius因担心Oracle收购影响MySQL的开源性,创建了MariaDB,提供额外功能和改进。维基百科、Google等已逐步替换为MariaDB,以确保更好的性能和社区支持。掌握MariaDB作为备用方案,对未来发展至关重要。
59 3
|
28天前
|
安全 关系型数据库 MySQL
MySQL崩溃保险箱:探秘Redo/Undo日志确保数据库安全无忧!
《MySQL崩溃保险箱:探秘Redo/Undo日志确保数据库安全无忧!》介绍了MySQL中的三种关键日志:二进制日志(Binary Log)、重做日志(Redo Log)和撤销日志(Undo Log)。这些日志确保了数据库的ACID特性,即原子性、一致性、隔离性和持久性。Redo Log记录数据页的物理修改,保证事务持久性;Undo Log记录事务的逆操作,支持回滚和多版本并发控制(MVCC)。文章还详细对比了InnoDB和MyISAM存储引擎在事务支持、锁定机制、并发性等方面的差异,强调了InnoDB在高并发和事务处理中的优势。通过这些机制,MySQL能够在事务执行、崩溃和恢复过程中保持
70 3
|
28天前
|
SQL 关系型数据库 MySQL
数据库灾难应对:MySQL误删除数据的救赎之道,技巧get起来!之binlog
《数据库灾难应对:MySQL误删除数据的救赎之道,技巧get起来!之binlog》介绍了如何利用MySQL的二进制日志(Binlog)恢复误删除的数据。主要内容包括: 1. **启用二进制日志**:在`my.cnf`中配置`log-bin`并重启MySQL服务。 2. **查看二进制日志文件**:使用`SHOW VARIABLES LIKE 'log_%';`和`SHOW MASTER STATUS;`命令获取当前日志文件及位置。 3. **创建数据备份**:确保在恢复前已有备份,以防意外。 4. **导出二进制日志为SQL语句**:使用`mysqlbinlog`
88 2
|
1月前
|
关系型数据库 MySQL 数据库
Python处理数据库:MySQL与SQLite详解 | python小知识
本文详细介绍了如何使用Python操作MySQL和SQLite数据库,包括安装必要的库、连接数据库、执行增删改查等基本操作,适合初学者快速上手。
265 15