MySQL锁机制

MySQL锁基础

锁的类型

1,MyISAM 存储引擎使用的是表级锁

2,InnoDB 存储引擎既支持行级锁,也支持表级锁,默认情况下使用行级锁。

3,所谓表级锁,它直接锁住的是一个表,开销小,加锁快,不会出现死锁的情况,锁定粒度大,发生锁冲突的概率更高,并发度最低。

4,所谓行级锁,它直接锁住的是一条记录,开销大,加锁慢,发生锁冲突的概率较低,并发度很高。

5,所谓页级锁,它是锁住的一个页面,在 InnoDB 中一个页面为16KB,它的开销介于表级锁和行级锁中间,也可能会出现死锁,锁定粒度也介于表级锁和行级 锁中间,并发度也介于表级锁和行级锁中间。

6,仅仅从锁的角度来说,表级锁更加适合于以查询为主的应用,只有少量按照索引条件更新数据的应用,比如大多数的 web 应用。

7,行级锁更适合大量按照索引条件并发更新少量不同的数据,同时还有并发查询的应用

innodb锁 

1,InnoDB 与 MyISAM 的相当大的两点不同在于:

(1.1) 支持事务

(1.2) 采用行级锁

2,行级锁本身与表级锁的实现差别就很大,而事务的引入也带来了很多新问题,尤其是事务的隔离性,与锁机制息息相关。

3,对于事务的基本操作,对于不同隔离级别可能引发的问题,像脏读、不可重复读等问题我们上一节就已经举例说明了,这里就不再赘述了。

4,数据库实现事务隔离的方式,基本可以分为两种:

(4.1) 在操纵数据之前,先对其加锁(排他锁),防止其他事务对数据进行修改。这就需要各个事务串行操作才可以实现。

select * from biao where id = 1 for update;
使事务串行化,解决超卖问题,但会降低性能

(4.2) 不加任何锁,通过生成一系列特定请求时间点的一致性数据快照,并通过这个快照来提供一致性读取。

select * from user where id =1; # 获取数据的同时,也获取当前数据版本
update user set username = '111',version = version+1 where id =1 and version=当前版本;
如果并发导致版本变化,那么这个时候数据更新是失败的,因为数据版本对不上

5,上面的(4.2)方式就是数据多版本并发控制,也就是多版本数据库,一般简称为 MVCC 或者 MCC,它是 Multi Version Concurrency Control 的简写。

6,数据库的事务隔离越严格,并发的副作用就越小,当然付出的代价也就越大,因为事务隔离机制实质上是使得事务在一定程度上”串行化”,这与并行是矛盾 的。

 innodb锁类型

1,InnoDB 实现了下面两种类型的锁:

(1)共享锁(S):允许一个事务去读一行,阻止其他事务获得相同数据集的排他锁。

(2)排他锁(X):允许获得排他锁的事务更新数据,阻止其他事务获得相同数据集的共享读锁和排他写锁。

2,这里有个锁兼容和冲突的概念,如果在加一个锁的时候,另一个锁可以加上去,那么就是锁兼容。如果加上一个锁之后,拒绝其他的锁加上,那么就是锁冲 突。

3,各种锁的兼容冲突情况如下:

(1)X 和所有锁都冲突

(2)IX 兼容 IX 和 IS

(3)S 兼容 S 和 IS

(4)IS 兼容 IS、IX 和 S

这个扩展需要参考其他的博客 深入理解MySQL锁 (S、X、IS、IX)RC模式下解析 - 墨天轮

4,如果一个事务请求的锁模式与当前的锁兼容,InnoDB 就将请求的锁授予该事务,如果两者是冲突的,那么该事务就要等待锁释放。

5,对于 update、delete、insert 语句,InnoDB 会自动给设计到的数据集加排他锁即 X。

6,对于 select 语句,InnoDB 不会加任何锁。

7,我们可以使用如下语句来显式的给数据集加锁:

(1)共享锁(S):select * from t1 where ... lock in share mode;

(2)排他锁(X):select * from t1 where ... for update;

8,我们可以用 select ...in share mode 来获得共享锁,主要用在数据依存关系时来确认某行记录是否存在,并确认没有人对这个记录进行 update 或者 delete 操 作。

9,我们可以使用 select... for update 来获得排他锁,它会拒绝其他事务在其上加其他锁。

 对于语句进行加锁

排它锁加锁语法:

--给`user`表id为1的数据加排它锁

start transaction--开启事务

select * from `user` where id=1 for update;--给id为1的数据加排它锁

commit;--提交事务

rollback;--回滚事务

共享锁加锁语法:

--给`user`表id为1的数据加共享锁

start transaction--开启事务

select * from `user` where id=1 lock in share mode;--给id为1的数据加共享锁

commit;--提交事务

rollback;--回滚事务

死锁的产生于处理 

死锁产生的原因

所谓死锁:是指两个或两个以上的进程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去.此时称系统处于死锁状态或系统产生了死锁,这些永远在互相等待的进程称为死锁进程。表级锁不会产生死锁.所以解决死锁主要还是针对于最常用的InnoDB。

死锁的关键在于:两个(或以上)的Session加锁的顺序不一致。

那么对应的解决死锁问题的关键就是:让不同的session加锁有次序

死锁的现象

事务1事务2分别对id为1与id为2的数据进行排它锁加锁,随后进行交叉的数据修改。

事务运行顺序事务1事务2
begin;begin;
select * from user where id = 1 for update;select * from user where id = 2 for update;
update user set username = 'shine York' where id =2;update user set username = 'starsky' where id =1;
commitcommit

 运行结果:

尽可能的避免事务死锁 

1)以固定的顺序访问表和行。比如对第2节两个job批量更新的情形,简单方法是对id列表先排序,后执行,这样就避免了交叉等待锁的情形;又比如对于3.1节的情形,将两个事务的sql顺序调整为一致,也能避免死锁。

2)大事务拆小。大事务更倾向于死锁,如果业务允许,将大事务拆小。

3)在同一个事务中,尽可能做到一次锁定所需要的所有资源,减少死锁概率。

4)降低隔离级别。如果业务允许,将隔离级别调低也是较好的选择,比如将隔离级别从RR调整为RC,可以避免掉很多因为gap锁造成的死锁。

5)为表添加合理的索引。可以看到如果不走索引将会为表的每一行记录添加上锁,死锁的概率大大增大。

乐观锁于悲观锁的解释

悲观锁(Pessimistic Lock), 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block直到它拿到锁。传统的关系型数据库里边就用到了很多这种锁机制,比如行锁,表锁等,读锁,写锁等,都是在做操作之前先上锁。

乐观锁(Optimistic Lock), 顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号等机制。乐观锁适用于多读的应用类型,这样可以提高吞吐量,像数据库如果提供类似于write_condition机制的其实都是提供的乐观锁。

乐观锁和悲观锁是一种思想,并不是MySQL的锁机制

MVCC-多版本并发控制实现乐观锁

MVCC本质

多版本并发控制 (Multiversion concurrency control, MCC 或 MVCC),是数据库管理系统常用的一种并发控制,也用于程序设计语言实现事务内存。

乐观并发控制和悲观并发控制都是通过延迟或者终止相应的事务来解决事务之间的竞争条件来保证事务的可串行化;虽然前面的两种并发控制机制确实能够从根本上解决并发事务的可串行化的问题,但是其实都是在解决写冲突的问题,两者区别在于对写冲突的乐观程度不同 (悲观锁也能解决读写冲突问题,但是性能就一般了)。而在实际使用过程中,数据库读请求是写请求的很多倍,我们如果能解决读写并发的问题的话,就能更大地提高数据库的读性能,而这就是多版本并发控制所能做到的事情。

与悲观并发控制和乐观并发控制不同的是,MVCC 是为了解决读写锁造成的多个、长时间的读操作饿死写操作问题,也就是解决读写冲突的问题。MVCC 可以与前两者中的任意一种机制结合使用,以提高数据库的读性能。

数据库的悲观锁基于提升并发性能的考虑,一般都同时实现了多版本并发控制。不仅是 MySQL,包括 Oracle、PostgreSQL 等其他数据库系统也都实现了 MVCC,但各自的实现机制不尽相同,因为 MVCC 没有一个统一的实现标准。

总的来说,MVCC 的出现就是数据库不满用悲观锁去解决读 - 写冲突问题,因性能不高而提出的解决方案。

实现方式

MVCC 的实现,是通过保存数据在某个时间点的快照来实现的。每个事务读到的数据项都是一个历史快照,被称为快照读,不同于当前读的是快照读读到的数据可能不是最新的,但是快照隔离能使得在整个事务看到的数据都是它启动时的数据状态。而写操作不覆盖已有数据项,而是创建一个新的版本,直至所在事务提交时才变为可见。

一张简化的商品表:

字段描述
id主键id
product_name商品名称
product_price商品价格
product_stock商品库存
version数据版本(新增默认为1)

 实现代码

<?php
try {
  $pdo=new pdo("mysql:host=localhost;dbname=mysql_php", "root", "root", array(PDO::ATTR_AUTOCOMMIT=>0));//最后是关闭自动提交
  $pdo->setAttribute(PDO::ATTR_ERRMODE,  PDO::ERRMODE_EXCEPTION);//开启异常处理
  try {
    $pdo->beginTransaction();//开启事务处理
    $product_id = 1;
    $sql = "select * from product where id = {$product_id}";
    $restful=$pdo->query($sql)->fetch(PDO::FETCH_ASSOC);
    if (!$restful) {
      echo "商品不存在:".$e->getMessage();
      exit;
    }else{
      $version = $restful['version'];
      $sql = "update product set product_stock=product_stock-1,version = version+1 where id = {$product_id} and version = {$version}";
      $restful=$pdo->exec($sql);
      if ($restful) {
        $pdo->commit();//事务提交
        echo "修改库存成功";
        exit;
      }else{
        echo "修改库存失败";
        exit;
      }
    }

  } catch (\Exception $e) {
    echo $sql.$e->getMessage();
    exit;
  }

} catch (\Exception $e) {
  echo "数据库连接失败:".$e->getMessage();
  exit;
}
?>

当前读和快照读

当前读

像 select lock in share mode (共享锁), select for update ; update, insert ,delete (排他锁) 这些操作都是一种当前读,为什么叫当前读?就是它读取的是记录的最新版本,读取时还要保证其他并发事务不能修改当前记录,会对读取的记录进行加锁。

快照读

像不加锁的 select 操作就是快照读,即不加锁的非阻塞读;快照读的前提是隔离级别不是未提交读和串行化级别,因为未提交读总是读取最新的数据行,而不是符合当前事务版本的数据行。而串行化则会对所有读取的行都加锁

优缺点

MVCC 使大多数读操作都可以不用加锁,这样设计使得读数据操作很简单,性能很好,并且也能保证只会读取到符合标准的行。不足之处是每行记录都需要额外的存储空间,需要做更多的行检查工作,以及一些额外的维护工作。

间隙锁与行锁升级为表锁

什么是间隙锁

当我们用范围条件而不是相等条件检索数据,并请求共享或排他锁时,InnoDB会给符合条件的已有数据记录的索引项加锁;对于键值在条件范围内但不存在的记录,叫做“间隙(GAP)”,InnoDB也会对这个“间隙”加锁,这种锁机制就是所谓的间隙锁(NEXT-KEY)锁。

危害:因为Query执行过程中通过范围查找的话,他会锁定整个范围内所有的索引键值,即使这个键值并不存在。间隙锁有一个比较致命的弱点,就是当锁定一个范围键值之后,即使某些不存在的键值也会被无辜的锁定,而造成在锁定的时候无法插入锁定值范围内的任何数据,在某些场景下这可能会针对性造成很大的危害。

行锁升级为表锁

众所周知,MySQL 的 InnoDB 存储引擎支持事务,支持行级锁(innodb的行锁是通过给索引项加锁实现的)。得益于这些特性,数据库支持高并发。如果 InnoDB 更新数据使用的不是行锁,而是表锁呢?是的,InnoDB 其实很容易就升级为表锁,届时并发性将大打折扣了。

常用的索引有三类:主键、唯一索引、普通索引。主键 不由分说,自带最高效的索引属性;唯一索引指的是该属性值重复率为0,一般可作为业务主键,例如学号;普通索引 与前者不同的是,属性值的重复率大于0,不能作为唯一指定条件,例如学生姓名。

在不使用索引的情况下进行加锁

不通过索引去锁数据,就会变成表锁!

索引失效的查询也会变成 表锁!

 在使用普通索引的情况进行加锁

在加了索引之后不再进行表锁,那是因为行锁是建立在索引字段的基础上,如果行锁定的列不是索引列则会升级为表锁

范围性查询测试

当要进行加锁的数据不确定时,也一样会是表锁。 

总结: 

行锁是建立在索引的基础上。

普通索引的数据重复率过高导致索引失效,行锁升级为表所

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值