Mysql中的innoDB如何解决幻读
目录
- 1.Mysql的事务隔离级别
- 2. 什么是幻读
- 3. InnoDB如何解决幻读的问题
- 4. 总结
1.Mysql的事务隔离级别
这四种隔离级别,当存在多个事务并发冲突的时候,可能会出现脏读,不可重复读,幻读的一些问题,而innoDB
在可重复读隔离级别模式下解决了幻读的一个问题,
2. 什么是幻读
幻读是指在同一个事务中,前后两次查询相同范围的时候得到的结果不一致
如图,第一个事务里面,我们执行一个范围查询,这个时候满足条件的数据只有一条,而在第二个事务里面,它插入一行数据并且进行了提交,接着第一个事务再去查询的时候,得到的结果比第一次查询的结果多出来一条数据,注意第一个事务的第一次和第二次查询,都在同一个事物里面,所以,幻读会带来数据一致性的问题
3. InnoDB如何解决幻读的问题
InnoDB引入间隙锁
和next-key lock
机制去解决幻读问题
假如现在存在这样一个B+Tree的索引结构,这个结构有4个索引元素,分别是1,4,7,10 当我们通过主键索引查询一条记录,并且对这条记录通过for update
加锁的时候
这个时候会产生一个记录锁,也就是行锁,锁定id=1
这个索引
被锁定的记录在锁释放之前,其他事务无法对这一条记录做任何操作的,前面我们所过对幻读的定义,幻读是指在同一个事务中,前后两次查询相同范围的时候得到的结果不一致,注意这里敲掉的是范围查询,也就是说要解决幻读的问题,必须保证一个点
就是如果一个事务通过这样一条语句进行锁定的时候,另外一个事务再执行
这样一条insert语句需要被阻塞,直到前面获得所的事务被释放,所以在innonDB设计一种间隙锁
,它的主要功能是锁定一定范围内的索引记录
当对查询范围id > 4 and id < 7
这个范围加锁的时候,会针对B+数中(4,7)这个开区间的范围加间隙锁
,意味着在这种情况下其他事务对这个区间的数据进行插入更新删除都会被锁住,但是还有另外一种情况,比如像这样
这条查询语句针对id > 4
这个条件加锁,那么它需要锁定多个索引区间,所以这个情况下InnoDB引入一个叫next-key lock
机制,next-key lock
相当于间隙锁和记录锁的合集,记录锁锁定存在记录的行,间隙锁锁住的是记录行之间的间隙,而next-key lock
锁住的是两者的和
每个数据行非唯一的索引列,都会存在一把next-key lock
,当某个事务持有这一行数据的next-key lock
的时候,会锁住一段在左开右闭区间的数据,因此当通过id > 4
这样一个范围加锁的时候,InnoDB会去加一个next-key lock
锁,锁定的区间范围是(4,7 ](7,10 ](10,+♾️ ]。间隙锁
和next-key lock
的区别是在加锁的范围,间隙锁
锁定的是两个索引之间的间隙,而next-key lock
会锁定多个索引区间,它包含记录锁
和间隙锁
当我们使用范围查询不仅仅命中Record记录,还包含了Gap间隙的时候,在这种情况下使用的就是临键锁,也就是next-key lock
它是Mysql里面默认的行锁算法
4. 总结
虽然InnoDB里面通过间隙锁
方式解决了幻读的问题但是加锁之后一定会影响到并发性能,因此对与性能较高的一些业务场景,我们可以把隔离级别设置不可重复,那么这个级别不存在间隙锁,也不存在性能的影响
相关推荐
-
MySQL的存储引擎InnoDB和MyISAM
目录 1.MyISAM底层存储 1.1MyISAM底层存储(非聚集索引方式) 1.2InnoDB底层存储(聚集索引方式) 2.InnoDB与MyISAM简介 3.MyISAM与InnoDB的比较 4.什么时候用MyISAM数据存储引擎?什么时候用InnoDB数据存储引擎? 1.MyISAM底层存储 (非聚集索引方式)与InnoDB底层存储(聚集索引方式) 1.1MyISAM底层存储(非聚集索引方式) Myisam 创建表后生成的文件有三个: frm:创建表的语句 MYD:表里面的数据文件(myi
-
MySQL InnoDB ReplicaSet(副本集)简单介绍
01 InnoDB ReplicaSet(副本集)介绍 在MySQL8.0引入了InnoDB ReplicaSet,它提供了我们熟悉的复制特性,如果有mongodb副本集的概念,理解InnoDB ReplicaSet会比较容易. InnoDB ReplicaSet使用了下面的技术: 1.MySQL Shell,一个功能更强大的客户端 2.MySQL Router,一个轻量级别的中间件,可以类比MongoDB中的mongos的角色: 3.MySQL Server,也就是MySQL服务 InnoDB
-
MySQL Innodb索引机制详细介绍
1.什么是索引 索引是存储引擎用于快速找到记录的一种数据结构. 2.索引有哪些数据结构 顺序查找结构:这种查找效率很低,复杂度为O(n).大数据量的时候查询效率很低. 有序的数据排列:二分查找法又称折半查找法. 通过一次比较,将查找区间缩小一半.而MySQL中的数据并不是有序的序列. 二叉查找树:左子树的键值总是小于根的键值,右子树的键值总是大于根的键值.通过中序遍历得到的序列是有序序列,但如果二叉查找树构造的不好则跟顺序查找没什么区别 平衡二叉树:如果需要二叉查找树是平衡的,从而引出平衡二叉树
-
详解MySql中InnoDB存储引擎中的各种锁
目录 什么是锁 InnoDB存储引擎中的锁 锁的算法 行锁的3种算法 幻像问题 锁的问题 脏读 不可重复读 丢失更新 死锁 什么是锁 现实生活中的锁是为了保护你的私有物品,在数据库中锁是为了解决资源争抢的问题,锁是数据库系统区别于文件系统的一个关键特性.锁机制用于管理对共享资源的并发访. 数据库系统使用锁是为了支持对共享资源进行并发访问,提供数据的完整性和一致性 InnoDB存储引擎区别于MyISAM的两个重要特征就是:InnoDB存储引擎支持事务和行级别的锁,MyISAM只支持表级别的锁 In
-
MySQL InnoDB 事务锁源码分析
目录 1. Lock 与 Latch 2. Repeatable Read 3. Insert加锁流程 3.1 lock mode 3.2 加锁流程 3.3 隐式锁 4. Select 加锁流程 本文前提: 代码MySQL 8.0.13 只整理Repeatable Read当前读.Read Committed简单很多,另外快照读是基于MVCC不用加锁,所以不在本文讨论范畴. 1. Lock 与 Latch InnoDB 中的lock是事务中对访问/修改的record加的锁,它一般是在事务提交或回
-
MySQL Innodb关键特性之插入缓冲(insert buffer)
什么是insert buffer? 插入缓冲,也称之为insert buffer,它是innodb存储引擎的关键特性之一,我们经常会理解插入缓冲时缓冲池的一个部分,这样的理解是片面的,insert buffer的信息一部分在内存中,另外一部分像数据页一样,存在于物理页中. 在innodb中,我们知道,如果一个表有自增主键,那么对于这个表的默认插入是非常快的,注意,这里的主键是自增的,如果不是自增的,那么这个插入将会变成随机的,就可能带来数据页分裂的开销,这样,插入就不是顺序的,就会变慢.还有一种
-
详解MySQL InnoDB存储引擎的内存管理
存储引擎之内存管理 在InnoDB存储引擎中,数据库中的缓冲池是通过LRU(Latest Recent Used,最近最少使用)算法来进行管理的,即最频繁使用的页在LRU列表的最前段,而最少使用的页在LRU列表的尾端,当缓冲池不能存放新读取到的页时,首先释放LRU列表尾端的页. 上面的图中,我使用8个数据页来表示队列,具体作用,先卖个关子.在InnoDB存储引擎中,缓冲池中页的默认大小是16KB,LRU列表中有一个midpoint的位置,新读取到的数据页并不是直接放入到LRU列表的首部,而是放入
-
Mysql中的innoDB如何解决幻读
目录 1.Mysql的事务隔离级别 2. 什么是幻读 3. InnoDB如何解决幻读的问题 4. 总结 1.Mysql的事务隔离级别 这四种隔离级别,当存在多个事务并发冲突的时候,可能会出现脏读,不可重复读,幻读的一些问题,而innoDB在可重复读隔离级别模式下解决了幻读的一个问题, 2. 什么是幻读 幻读是指在同一个事务中,前后两次查询相同范围的时候得到的结果不一致 如图,第一个事务里面,我们执行一个范围查询,这个时候满足条件的数据只有一条,而在第二个事务里面,它插入一行数据并且进行了提交,接
-
MySQL可重复读级别能够解决幻读吗
引言 之前在深入了解数据库理论的时候,了解到事物的不同隔离级别可能存在的问题.为了更好的理解所以在MySQL数据库中测试复现这些问题.关于脏读和不可重复读在相应的隔离级别下都很容易的复现了.但是对于幻读,我发现在可重复读的隔离级别下没有出现,当时想到难道是MySQL对幻读做了什么处理? 测试: 创建一张测试用的表dept: CREATE TABLE `dept` ( `id` int(11) NOT NULL AUTO_INCREMENT, `name` varchar(20) DEFAULT
-
详解MySQL是如何解决幻读的
一.什么是幻读 在一次事务里面,多次查询之后,结果集的个数不一致的情况叫做幻读. 而多出来或者少的哪一行被叫做 幻行 二.为什么要解决幻读 在高并发数据库系统中,需要保证事务与事务之间的隔离性,还有事务本身的一致性. 三.MySQL 是如何解决幻读的 如果你看到了这篇文章,那么我会默认你了解了 脏读 .不可重复读与可重复读. 1. 多版本并发控制(MVCC)(快照读) 多数数据库都实现了多版本并发控制,并且都是靠保存数据快照来实现的. 以 InnoDB 为例,每一行中都冗余了两个字断.一个是行的
-
MySQL如何解决幻读问题
目录 前言 一.什么是幻读? 二.幻读有什么问题? (1)需要单独解决 (2)间隙锁引发的并发度 三.如何解决幻读? 三.总结 前言 我们知道MySQL在可重复读隔离级别下别的事物提交的内容,是看不到的.而可提交隔离级别下是可以看到别的事务提交的.而如果我们的业务场景是在事物内同样的两个查询我们需要看到的数据都是一致的,不能被别的事物影响,就使用可重复读隔离级别.这种情况下RR级别下的普通查询(快照读)依靠MVCC解决"幻读"问题,如果是"当前读"的情况需要依靠什么
-
Mysql事务的隔离级别(脏读+幻读+可重复读)
目录 一.事务 (一)什么是事务 (二)事务的特征 (三)手动开启.提交.回滚事务 二.脏读.幻读.可重复读 (一)脏读 read uncommit (二)不可重复读 unrepeatable (三)幻读 (四)不可重复读和幻读的区别 三.事务的隔离级别 (一)读未提交read uncommitted (二)读已提交read committed (三)不可重复读 read repeatable (四)序列化 serializable 前言: 因为InnoDB是支持事务的,所以只对InnoDB的事
-
vs如何读取mysql中的数据并解决中文乱码问题
一.在vs中新建一个工程,然后新建一个源文件 二.右击上述图片中的mysql_test工程名,然后在最下方找到属性,并点击 三.点击VC++目录,在包含目录中添加mysql的文件路径,一般情况下的mysql的默认安装路径都是在C盘的Program Files下的,找到include路径,然后点击确定 四.在C/C++中的常规选项中,找到附加包含目录,同时也将mysql的include路径包含进去 五.在属性页中选择链接器的常规选项,在里面找到附加库目录,然后添加mysql的lib路径 六. 在
-
MySQL不支持InnoDB的解决方法
G一下后,解决如下: /var/lib/mysql目录下,删除ibdata1.ib_logfile1. ib_logfile0,然后重启MySql让其重建以上文件: mysqladmin -uroot -p shutdown sudo mysqld_safe & 搞定! 下面是网络上的其它文章.大家也可以参考下.早上起来,到PHP站点去看了下,准备测试下别人写的一个CMS系统,高兴的下载了程序,然后把程序拷贝到所在目录.由于该程序没有install.php,里面只包含了一个*.sql的数据库语句
-
mysql中engine=innodb和engine=myisam的区别介绍
最开始用MySQL Administrator建数据库的时候,表缺省是InnoDB类型,也就没有在意.后来用Access2MySQL导数据的时候发现只能导成 MyISAM类型的表,不知道这两种类型有什么区别,就去查了查.原来是MyISAM类型不支持事务处理等高级处理,而InnoDB类型支持. MyISAM类型的表强调的是性能,其执行数度比InnoDB类型更快,但是不提供事务支持,而InnoDB提供事务支持已经外部键等高级数据库功能.这 样就可以根据数据表不同的用处是用不同的存储类型. 另外,My
-
MySQL中Innodb的事务隔离级别和锁的关系的讲解教程
前言: 我们都知道事务的几种性质,数据库为了维护这些性质,尤其是一致性和隔离性,一般使用加锁这种方式.同时数据库又是个高并发的应用,同一时间会有大量的并发访问,如果加锁过度,会极大的降低并发处理能力.所以对于加锁的处理,可以说就是数据库对于事务处理的精髓所在.这里通过分析MySQL中InnoDB引擎的加锁机制,来抛砖引玉,让读者更好的理解,在事务处理中数据库到底做了什么. 一次封锁or两段锁? 因为有大量的并发访问,为了预防死锁,一般应用中推荐使用一次封锁法,就是在方法的开始阶段,已经预先知道会
-
详解MySQL 幻读及如何消除
这是一篇数据库隔离级别的科普文章,旨在了解数据库中著名的幻读现象,为了专注,对脏读.不可重复读不作讨论. 事务隔离级别 MySQL有四级事务隔离级别: 读未提交 READ-UNCOMMITTED: 存在脏读,不可重复读,幻读的问题 读已提交 READ-COMMITTED:不存在脏读,但存在不可重复读,幻读问题 可重复读 REPEATABLE-READ:不存在脏读,不可重复读问题,但存在幻读问题 序列化SERIALIZABLE:解决脏读,不可重复读,幻读问题,但完全串行执行,性能最低 什么是幻读
随机推荐
- Android输入框添加emoje表情图标的实现代码
- spring boot启动加载数据原理分析
- Oracle date 和 timestamp 区别详解
- 进度条在.net导入Excel时的应用实例
- 一个用于网络的工具函数库
- MySQL 5.7.9 服务无法启动-“NET HELPMSG 3534”的解决方法
- SQL Server 数据库分离与附加 就这么简单!
- 使用python编写批量卸载手机中安装的android应用脚本
- 网络漏洞扫描系统必要性
- Android计时器控件Chronometer应用实例
- 基于Listener监听器生命周期(详解)
- C#自定义的字符串操作增强类实例
- C#判断ip地址是否可以ping的通
- Android自定义View实现拖动选择按钮
- Android 根据EditText搜索框ListView动态显示数据
- 快速处理vue渲染前的显示问题
- vue element-ui table组件动态生成表头和数据并修改单元格格式 父子组件通信
- mysql下centos安装mariadb的方法及遇到的问题
- PHP模版引擎原理、定义与用法实例
- PHP微信支付结果通知与回调策略分析