如何保证数据并发访问的一致性、有效性是所有数据库必须解决的一个问题,锁冲突也是影响数据库并发访问性能的一个重要因素。从这个角度来说,锁对数据库而言显得尤其重要,也更加复杂。

表锁和行锁

mysql最显著的特点是不同的存储引擎支持不同的锁机制。比如,

  • MyISAM和MEMORY存储引擎采用的是表级锁。
  • InnoDB存储引擎既支持行级锁也支持表级锁,但默认情况下是采用行级锁。

读锁和写锁

  • 读锁(共享锁):当一个表或一行数据被加上读锁,则其他申请读锁操作将被允许,但其他申请写锁操作将被阻塞,直到锁被释放。
  • 写锁(独占锁):当一个表或一行数据被加上写锁,则其他申请读锁操作和申请写锁操作都将被阻塞,直到锁被释放。

一、MyISAM:

首先我们创建两个测试表及表数据:

create table tb_myISAM1(
id int auto_increment primary key,
value varchar(50) null
) engine=MyISAM; create table tb_myISAM2(
id int auto_increment primary key,
value varchar(50) null
) engine=MyISAM; insert into tb_myISAM1(value) values (uuid());
insert into tb_myISAM1(value) values (uuid()); insert into tb_myISAM2(value) values (uuid());
insert into tb_myISAM2(value) values (uuid());

在执行查询语句(SELECT)前,会自动给涉及的所有表加读锁,在执行更新操作(UPDATE、DELETE、INSERT等)前,会自动给涉及的表加写锁,这个过程并不需要用户干预。所以我们采用通过sql脚本显式加锁的方式进行测试。

(一)表读锁

先说结论:如果某个session通过脚本显式的给表增加了读锁,那么

  • 当前session:可以针对该表执行读操作;不可以针对该表执行写操作(会异常);不可以针对其他表执行读操作(会异常);不可以针对其他表执行写操作(会异常)。
  • 其他session:可以针对该表执行读操作;不可以针对该表执行写操作(会阻塞);可以针对其他表执行读操作;可以针对其他表执行写操作。

验证脚本:

lock tables tb_myISAM1 read;  --
select * from tb_myISAM1; --
insert into tb_myISAM1 (value) values (uuid()); --
select * from tb_myISAM2; --
insert into tb_myISAM2 (value) values (uuid()); --
unlock tables; --

我们为上面的脚本的每一行增加了序号以表述方便:

建立session1,执行1后:执行2通过,执行3异常,执行4异常,执行5异常;不要关闭session1再建立session2:执行2通过,执行3阻塞,执行4通过,执行5通过。

(二)表写锁

先说结论:如果某个session通过脚本显式的给表增加了写锁,那么

  • 当前session:可以针对该表执行读操作;可以针对该表执行写操作;不可以针对其他表执行读操作(会异常);不可以针对其他表执行写操作(会异常)。
  • 其他session:不可以针对该表执行读操作(会阻塞);不可以针对该表执行写操作(会阻塞);可以针对其他表执行读操作;可以针对其他表执行写操作。

验证脚本:

lock tables tb_myISAM1 write;  --
select * from tb_myISAM1; --
insert into tb_myISAM1 (value) values (uuid()); --
select * from tb_myISAM2; --
insert into tb_myISAM2 (value) values (uuid()); --
unlock tables; --

我们为上面的脚本的每一行增加了序号以表述方便:

建立session1,执行1后:执行2通过,执行3通过,执行4异常,执行5异常;不要关闭session1再建立session2:执行2阻塞,执行3阻塞,执行4通过,执行5通过。

(三)锁争抢

如果有两个session同时申请同一个表的不同锁,session1申请读锁,session2申请写锁,那么MyISAM会先执行写锁,哪怕是申请读锁操作先到。这是因为MyISAM认为写请求一般比读请求要重要。这也正是MyISAM表不太适合于有大量更新操作操作应用的原因,因为,大量的更新操作会造成查询操作很难获得读锁,从而可能永远阻塞。

二、innoDB

innoDB与MyISAM的最大不同有两点:一是支持事务;二是采用了行级锁。在mysql的事务中,对表或行的加锁解锁操作时自动执行的,一般不需要用户干预。但是用户可以在start transaction开启事务后,可通过脚本显式的进行加锁和解锁,事务commit或rollback后锁会自动释放。

-- 行读锁例子
select * from dt_table1 where id = 1 lock in share mode;
-- 行写锁例子
select * from dt_table1 where id = 1 for update;

首先我们创建两个测试表及表数据

create table dt_table1(
id int auto_increment primary key,
value varchar(50) null
) ;insert into dt_table1(value) values (uuid());
insert into dt_table1(value) values (uuid());

(1)行读锁

先说结论:如果某个session为某一行增加了读锁,那么:

  1. 当前session:可以重新针对该行加写锁;可以针对该行执行读操作;可以针对该行执行写操作。
  2. 其他session:可以针对该行加读锁;不可以针对该行加写锁(会阻塞),可以针对该行执行读操作;不可以针对该行执行写操作(会阻塞)。

验证脚本:

start transaction; --
select * from dt_table1 where id = 1 lock in share mode; --
select * from dt_table1 where id = 1 for update ; --
select * from dt_table1 where id = 1; --
update dt_table1 set value = '' where id = 1; --
commit; --

我们为上面的脚本的每一行增加了序号以表述方便:

建立session1执行1、2后:执行3通过,执行4通过,执行5通过;不要关闭session1再建立session2并执行1后:执行2通过,执行3阻塞,执行4通过,执行5阻塞。

(2)行写锁

先说结论:如果某个session为某一行增加了写锁,那么:

  1. 当前session:可以重新针对该行加读锁;可以针对该行执行读操作;可以针对该行执行写操作。
  2. 其他session:不可以针对该行加读锁(会阻塞);不可以针对该行加写锁(会阻塞);不可以针对该行执行读操作(会阻塞);不可以针对该行执行写操作(会阻塞)。

验证脚本:

start transaction; --
select * from dt_table1 where id = 1 lock in share mode; --
select * from dt_table1 where id = 1 for update ; --
select * from dt_table1 where id = 1; --
update dt_table1 set value = '' where id = 1; --
commit; --

我们为上面的脚本的每一行增加了序号以表述方便:

建立session1执行1、3后:执行2通过,执行4通过,执行5通过;不要关闭session1再建立session2并执行1后:执行2阻塞,执行3阻塞,执行4阻塞,执行5阻塞。

(3)行锁机制

InnoDB行锁是通过给索引上的索引项加锁来实现的,所以在上例中我们创建表dt_table1的时候,id是作为主键出现的。这一点和oracle不同,oracle是通过在数据块中对相应数据行加锁来实现的。所以InnoDB这种行锁实现特点意味着:

  1. 只有通过索引条件检索数据,InnoDB才使用行级锁,否则,InnoDB将使用表锁。
  2. 如果是使用相同的索引键,是会出现锁冲突的。

我们来验证第一条:

创建测试及表数据:

create table dt_table2(
id int,
value varchar(50) null
) ; insert into dt_table2(id, value) values (1, uuid());
insert into dt_table2(id, value) values (2, uuid());

建立session1,执行

start transaction;
select * from dt_table2 where id = 1 for update;

建立session2,执行

start transaction;
update dt_table2 set value = '' where id = 2;

会发现session2中执行的update操作被阻塞了,尽管session1中加写锁的行和session2中修改的行并不是同一行。这就是因为id字段没有索引,在session1中给id=1加写锁的时候,实际上是加的表写锁,而不是行写锁,所以才导致session2中的update操作阻塞。

我们来验证第二条:

创建测试及表数据:

create table dt_table3(
id1 int,
id2 int,
value varchar(50) null
) ; create index idx_dt_table3 ON dt_table3(id1); insert into dt_table3(id1, id2, value) values (1, 1, uuid());
insert into dt_table3(id1, id2, value) values (1, 2, uuid());
insert into dt_table3(id1, id2, value) values (2, 1, uuid());
insert into dt_table3(id1, id2, value) values (2, 2, uuid());

建立session1,执行

start transaction;
select * from dt_table3 where id1 = 1 and id2 = 1 for update;

建立session2,执行如下脚本,update操作会被阻塞

start transaction;
update dt_table3 set value = '' where id1 = 1 and id2 = 2;

建立session3,执行如下脚本,执行成功

start transaction;
update dt_table3 set value = '' where id1 = 2 and id2 = 1;

(4)未完待续

mysql锁探究和实验的更多相关文章

  1. mysql 锁 事务隔离级别

    主题 最近在看mysql相关的书籍.实验了一些内容.分享一下,主要是关于事务隔离级别(read-committed和repeatable-read)和锁相关的. 很多网上文章上都能搜索到 read-c ...

  2. [MySQL] 锁/死锁问题一例

    MySQL锁/死锁问题 在MySQL中, 不同事务隔离级别下, 锁的情况表现是不同的, 另外表的设计上有无索引也是一个因素. 做一个小的实验测试InnoDB锁表现 -:) 说明 事务隔离级别 READ ...

  3. MySQL 锁(完整版)

    目录 锁总览 锁的作用 加锁流程 锁对数据库的影响 锁等待 死锁 锁类型 锁范围 锁方式 全局锁 全局读锁 全局QC锁 QC锁存在的问题: 备份锁 backup lock MDL锁 MDL锁类型 MD ...

  4. mysql锁

    锁是计算机协调多个进程或线程并发访问某一资源的机制.在数据库中,除传统的计算资源(如CPU.RAM.I/O等)的争用以外,数据也是一种供许多用户共享的资源.如何保证数据并发访问的一致性.有效性是所有数 ...

  5. Mysql锁初步

    存储引擎 要了解mysql的锁,就要先从存储引擎说起. 常用存储引擎列表如下图所示: 最常使用的两种存储引擎: Myisam是Mysql的默认存储引擎.当create创建新表时,未指定新表的存储引擎时 ...

  6. mysql锁表机制及相关优化

    (该文章为方便自己查阅,也希望对大家有所帮助,转载于互联网) 1. 锁机制 当前MySQL支持 ISAM, MyISAM, MEMORY (HEAP) 类型表的表级锁,BDB 表支持页级锁,InnoD ...

  7. MySQL锁系列3 MDL锁

    http://www.cnblogs.com/xpchild/p/3790139.html   MySQL为了保护数据字典元数据,使用了metadata lock,即MDL锁,保证在并发的情况下,结构 ...

  8. 01 MySQL锁概述

    锁是计算机协调多个进程或线程并发访问某一资源的机制.在数据库中,除传统的计算资源(如CPU.RAM.I/O 等)的争用以外,数据也是一种供许多用户共享的资源.如何保证数据并发访问的一致性.有效性是所有 ...

  9. Mysql锁机制介绍

    Mysql锁机制介绍 一.概况MySQL的锁机制比较简单,其最显著的特点是不同的存储引擎支持不同的锁机制.比如,MyISAM和MEMORY存储引擎采用的是表级锁(table-level locking ...

随机推荐

  1. CodeForces - 869C The Intriguing Obsession(组合数)

    题意:有三个集合,分别含有a.b.c个点,要求给这些点连线,也可以全都不连,每两点距离为1,在同一集合的两点最短距离至少为3的条件下,问有多少种连接方案. 分析: 1.先研究两个集合,若每两个集合都保 ...

  2. java内部类 2.19

    1.定义了成员内部类后,必须使用外部类对象来创建内部类对象,而不能直接去 new 一个内部类对象,即:内部类 对象名 = 外部类对象.new 内部类( ); //外部类HelloWorld publi ...

  3. 原生JS 实现 dom ready

    记录一下项目技术问题: 记得:放在head标签内的脚本,第一时间执行 var baseTools = { // dom ready ready: function( f ){ var ie = !!( ...

  4. 利用ZXing.Net生成和识别二维码

    ZXing.Net:ZXing是一个开放源码的,用Java实现的多种格式的1D/2D条码图像处理库. github:https://github.com/micjahn/ZXing.Net 直接将字符 ...

  5. DataFoundation比赛总结

    2018.3.20号左右,因为研究生的数据挖掘课程的老师要求我们集体参加一个比赛 ,所以在比赛参与时间.比赛难度和比赛类型的几种条件下,我们选择了2018平安产险数据建模大赛-驾驶行为预测驾驶风险比赛 ...

  6. TS文件极简合并

    TS文件是可以直接通过二进制拷贝连接的方式进行合并的,一般采用如下的命令行参数:copy /b 1.ts+2.ts+3.ts new.ts 这个例子就是将1.ts.2.ts.3.ts三个文件按顺序连接 ...

  7. 在 Windows 系统上安装 Jekyll

    目录 安装 Ruby 环境 用 Bundler 安装 Jekyll 本文是写给完全未用过 Ruby 乃至命令行工具者的.对于一般的开发者,Jekyll 官方文档的相关内容已然足够. 本文为钱院学辅技术 ...

  8. StackExchange.Redis.DLL 操作redis加强版

    直接引用StackExchange.Redis.dll这一个dll来操作redis App.config配置 <?xml version="1.0" encoding=&qu ...

  9. Java中调试与测试常用方法

    一.断言 1.概述 根据之前的条件来对后续的结果进行预判. 2.在Java中,断言默认是不开启的,需要手动进行开启,只需添加参数-ea(enable assert的意思) 开启步骤: 选中项目路径,鼠 ...

  10. SciKit-Learn 数据集基本信息

    ## 保留版权所有,转帖注明出处 章节 SciKit-Learn 加载数据集 SciKit-Learn 数据集基本信息 SciKit-Learn 使用matplotlib可视化数据 SciKit-Le ...