【Gap锁】Mysql的Gap锁在中文列下间隙怎样确定?

通过本文记录一次Gaplock的验证,网上大多gaplock是基于明确是数字型列来测试gaplock的,这里不再重复,随便贴个相关地址:https://www.cnblogs.com/crazylqy/p/7821481.html

我的疑问是如果所在的列是非数字型的列,比如说是列值是中文的话,mysql是如何确定间隙的,下面开始进行验证。

1、查看mysql间隙锁是否开启,默认"OFF"表示开启  => show variables like ‘innodb_locks_unsafe_for_binlog‘;

2、准备一张表,3个字段分别是 => id(主键索引),name,key(普通索引)    ,    然后随便搞几条数据。

CREATE TABLE `test_gaplock` (
`id` varchar(32) CHARACTER SET utf8 NOT NULL,
`name` varchar(255) CHARACTER SET utf8 DEFAULT NULL,
`key` varchar(32) COLLATE utf8_bin DEFAULT NULL,
PRIMARY KEY (`id`),
KEY `idx_key` (`key`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin;

3、通过mysql的客户端工具开3个会话窗口

  1)目标1:验证按照name列(无索引)更新的gaplock问题。

    结论:name列无索引作为where条件更新会锁表(全表GAP锁,这里因为我对锁的理解还不透彻,所以才多做了这次试验)

  2)  目标2:验证按照id列(主键索引)更新的gaplock问题。

    结论:这里我也懒得验证了,mysql的next-key lock(gap lock+record lock)机制就是,对于任何的当前读(如果对快照度和当前读有疑惑,请自行百度)操作,产不产生间隙锁取决于该列的索引是否能唯一确定一行数据(主键索引或者唯一索引),如果是,会升级成记录锁,也即普通行锁

  3)目标3:验证按照key列(普通索引),理论上普通索引会产生gap锁,key列现在存的是中文,他的间隙是怎么确定的,是按照自然排序(mysql默认按照主键排序),还是where的筛选列排序来确定间 隙?这里要直接引入一个结论就是,插入间隙区间是左闭合,又开启的区间即这种 [ ) 。意思就是比如你的列值是2,4,6,然后你更新的条件是4,那么你锁住的间隙是[2,6) ,操作2会被锁住,操作6是不受影响的。大家可自行验证,欢迎纠正。

先插入这么一条数据 => insert into test_gablock values(‘6‘,‘a4‘,‘李‘); 然后查看两种排序的展示情况

(下面的图表示自然排序)

(下面的图表示按照key列排序)

删除掉这行新增的记录,这里有个猜想就是要么是按照主键来排序,然后确定间隙,要么是按照筛选列来排序,确定间隙,根据上面的排序,更新name为a3,key为"强"的记录。

情况1:若是按照自然排序,那么间隙应该是[广,马) ,

情况2:若是按照key列排序,那么间隙应该是[广,陈) 。

然后现在再插入这条记录 => insert into test_gablock values(‘6‘,‘a4‘,‘李‘);

按照上面那个直接引入的结论,如果是情况1,那么是不会阻塞的,如果是情况2,那么是会阻塞的。

--------------------会话1(开启事务后不提交锁住记录)---------------------

set autocommit=0; 
begin;
update test_gaplock t1 set t1.`name` = a3‘ where t1.key = ‘强‘;

--------------------会话2(插入后不提交直接回滚,只是不想把验证数据弄坏而已)---------------------

set autocommit=0;  
begin;
insert into test_gaplock values(‘6‘,‘a4‘,‘李‘);
rollback;

--------------------会话3(只是用来方便查看实时数据)---------------------

select * from test_gaplock  t1 order by t1.name

结论:经过验证执行 => insert into test_gablock values(‘6‘,‘a4‘,‘李‘); 命令会被阻塞,所以情况2是正确的

原文地址:https://www.cnblogs.com/chengqqqq/p/12021217.html

时间: 2024-10-10 20:52:11

【Gap锁】Mysql的Gap锁在中文列下间隙怎样确定?的相关文章

Mysql 的表级锁和行级锁

表级锁 MySQL表级锁分为读锁和写锁. 读锁 用法:LOCK TABLE table_name [ AS alias_name ] READ 释放锁使用UNLOCK tables.可以为表使用别名,如果一旦使用别名在使用的时候也必须采用别名.成功申请读锁的前提是当前没有线程对该表使用写锁,否则该语句会被阻塞.申请读锁成功后,其他线程也可以对该表进行读操作,但不允许有线程对其进行写操作,就算是当前线程也不允许.当锁住了A表之后,就只能对A表进行读操作,对其他表进行读操作会出现错误(tablena

MySQL InnoDB锁机制之Gap Lock、Next-Key Lock、Record Lock解析

MySQL InnoDB支持三种行锁定方式: l   行锁(Record Lock):锁直接加在索引记录上面,锁住的是key. l   间隙锁(Gap Lock):锁定索引记录间隙,确保索引记录的间隙不变.间隙锁是针对事务隔离级别为可重复读或以上级别而已的. l   Next-Key Lock :行锁和间隙锁组合起来就叫Next-Key Lock. 默认情况下,InnoDB工作在可重复读隔离级别下,并且会以Next-Key Lock的方式对数据行进行加锁,这样可以有效防止幻读的发生.Next-K

Mysql中那些锁机制之InnoDB

我们知道mysql在曾经.存储引擎默认是MyISAM.可是随着对事务和并发的要求越来越高,便引入了InnoDB引擎.它具有支持事务安全等一系列特性. InnoDB锁模式 InnoDB实现了两种类型的行锁. 共享锁(S):同意一个事务去读一行,阻止其它事务获得同样的数据集的排他锁. 排他锁(X):同意获得排他锁的事务更新数据,可是组织其它事务获得同样数据集的共享锁和排他锁. 能够这么理解: 共享锁就是我读的时候,你能够读,可是不能写.排他锁就是我写的时候.你不能读也不能写.事实上就是MyISAM的

巧用MySQL InnoDB引擎锁机制解决死锁问题(转)

该文会通过一个实际例子中的死锁问题的解决过程,进一步解释innodb的行锁机制 最近,在项目开发过程中,碰到了数据库死锁问题,在解决问题的过程中,笔者对MySQL InnoDB引擎锁机制的理解逐步加深. 案例如下: 在使用Show innodb status检查引擎状态时,发现了死锁问题: *** (1) TRANSACTION: TRANSACTION 0 677833455, ACTIVE 0 sec, process no 11393, OS thread id 278546 starti

mysql事务和锁InnoDB(转)

背景 MySQL/InnoDB的加锁分析,一直是一个比较困难的话题.我在工作过程中,经常会有同事咨询这方面的问题.同时,微博上也经常会收到MySQL锁相关的私信,让我帮助解决一些死锁的问题.本文,准备就MySQL/InnoDB的加锁问题,展开较为深入的分析与讨论,主要是介绍一种思路,运用此思路,拿到任何一条SQL语句,都能完整的分析出这条语句会加什么锁?会有什么样的使用风险?甚至是分析线上的一个死锁场景,了解死锁产生的原因. 注:MySQL是一个支持插件式存储引擎的数据库系统.本文下面的所有介绍

mysql 中的 latch锁和Tlock(事务锁), DML加锁规则,以及死锁分析。

一.Latch和Tlock的关系 Latch:为保护临界资源的正确性而设计,例如保护正在使用的内存页面不被破坏等. 没有死锁检测机制,轻量锁,并且作用对象时内存页面或是内存共享变量. Tlock:事务锁,作用对象是事务,有死锁检测机制. 在innodb内部,为了减少死锁的发生概率,Latch不会等待Tlock. 线程获取行锁的流程: 在对行加锁的时候会先对行所在的页面添加lath,然后再对行添加Tlock,待对行添加完Tlock后再释放页面的Lath. 这种机制主要是为了保证线程获取的行数据的一

MySQL中的锁(表锁、行锁)

锁是计算机协调多个进程或纯线程并发访问某一资源的机制.在数据库中,除传统的计算资源(CPU.RAM.I/O)的争用以外,数据也是一种供许多用户共享的资源.如何保证数据并发访问的一致性.有效性是所在有数据库必须解决的一个问题,锁冲突也是影响数据库并发访问性能的一个重要因素.从这个角度来说,锁对数据库而言显得尤其重要,也更加复杂. 概述 相对其他数据库而言,MySQL的锁机制比较简单,其最显著的特点是不同的存储引擎支持不同的锁机制. MySQL大致可归纳为以下3种锁: 表级锁:开销小,加锁快:不会出

MySQL- InnoDB锁机制

InnoDB与MyISAM的最大不同有两点:一是支持事务(TRANSACTION):二是采用了行级锁.行级锁与表级锁本来就有许多不同之处,另外,事务的引入也带来了一些新问题.下面我们先介绍一点背景知识,然后详细讨论InnoDB的锁问题. 背景知识 事务(Transaction)及其ACID属性 事务是由一组SQL语句组成的逻辑处理单元,事务具有以下4个属性,通常简称为事务的ACID属性. 原子性(Atomicity):事务是一个原子操作单元,其对数据的修改,要么全都执行,要么全都不执行. 一致性

mysql事务和锁InnoDB

背景 MySQL/InnoDB的加锁分析,一直是一个比较困难的话题.我在工作过程中,经常会有同事咨询这方面的问题.同时,微博上也经常会收到MySQL锁相关的私信,让我帮助解决一些死锁的问题.本文,准备就MySQL/InnoDB的加锁问题,展开较为深入的分析与讨论,主要是介绍一种思路,运用此思路,拿到任何一条SQL语句,都能完整的分析出这条语句会加什么锁?会有什么样的使用风险?甚至是分析线上的一个死锁场景,了解死锁产生的原因. 注:MySQL是一个支持插件式存储引擎的数据库系统.本文下面的所有介绍