本文主要记录InnoDB存储引擎中锁的关键点,下篇文章通过实例确认加锁的范围。

InnoDB中的锁

1. 锁提供数据完整性和一致性

2. InnoDB行级锁:共享锁(S)和排他锁(X)。

  为了支持多粒度锁定,InnoDB支持意向锁,该锁允许事务在行锁和表锁同时存在。包括意向共享锁(IS,为某些记录加意向共享锁)和意向排他锁(IX,为某些记录加意向排他锁)

  意向锁将锁定的对象分为多个层次,意味着事务希望在更细粒度上进行加锁,如需要对页上的记录r加X锁,分别需要对数据库、表、页加意向锁IX,最后对记录r加X锁,其中任何一部分导致等待,该操作需要等待粗粒度锁的完成。

3. 锁的查看方式

  通过show engine innodb status来查看,其中的transactions片段可以看到事务,其中包括锁等待。

  在information_schema架构下,有3个表记录了事务和锁相关的信息。分别是INNODB_TRX,INNODB_LOCKS,INNODB_LOCK_WAITS。(具体看书或博客)

4. 一致性非锁读

  非锁定读机制,是InnoDB存储引擎的默认设置,默认读取不会占用和等待表上的锁

  InnoDB存储引擎利用行多版本控制实现一致性非锁读,

  当读取的行正在加X锁DELETE或UPDATE时,读操作不会等待锁释放,会读取行的一个快照数据

  快照数据是指该行之前版本的数据,其通过undo段完成的,因为undo用于事务中回滚数据,因此快照数据没有额外的开销,并且读快照也不需要加锁

  每行记录可能有多个版本的快照,由此带来的并发控制,称为多版本并发控制(Multi Version Concurrency Control,MVCC)

  在READ COMMIT事务隔离级别下,非锁一致性读总是读取被锁定行的最新一份快照数据,故可能会读到其他事务的提交,违反了ACID的隔离性

  在REPEATALE READ事务隔离级别下,非锁一致性读总是读取事务开始时的行数据版本

5. 一致性锁定读

  除了默认非锁定一致性读,还可以显式加锁读,有2种一致性锁定读:

    SELECT .... FOR UPDATE    加IX锁

    SELECT .... LOCK INSHARE MODE     加IS锁

6. 自增长与锁

  对含有自增长值的表都有一个自增长计数器,插入时根据该自增长计数器加1赋予该增长列,通过AUTO-INC Locking实现,其为特殊的表锁机制

  并发插入时,该方式性能较差,一个事务需要等待另一个事务而阻塞

  5.1.22版本后,InnoDB提供了一种轻量级互斥量的自增长实现机制,提高插入性能

  InnoDB通过innodb_autoinc_lock_mode参数控制自增长模式,值有0\1\2

  自增长值的列必须是索引,同时必须是索引的第一个列

7. 外键和锁

  对一个外键列,如果没有显式对该列加索引,InnoDB会自动为其加一个索引,可以避免表锁(锁是对索引进行锁定的)

  对于外键值的插入或更新是首先需要查询父表中的记录,即SELECT 操作,会为父表的记录使用SELECT....LOCK IN SHARE MODE加一个S锁。而不是使用一致性非锁定读的方式,因为该方式在读取时可能读到的是行记录的快照,如果父表同时更改该记录,可能会发生数据不一致的问题

  如事务A中父表删除记录r(加X锁,未提交),事务B插入引用父表记录r的行

    若在父表加S锁,碰到父表已经加了X锁,则事务B阻塞等待;

    若使用一致性非锁定读,并在REPEATABLE READ事务隔离级别下,可以读到父表中的r记录,事务B插入成功。

    等到事务A提交后,就会出现父、子表数据不一致的情况。

锁的算法

1. InnoDB支持行级锁,还支持范围锁。有3种行锁实现:

  Record Lock:单行锁,锁住索引记录,如果表没有建立索引,会使用隐式的主键进行锁定

  Gap Lock:间隙锁,锁定一个范围,不包括记录本身。可由不同的事务共同持有,只是阻止插入(MySQL文档描述:Gap locks in InnoDB are “purely inhibitive”, which means that their only purpose is to prevent other transactions from inserting to the gap. Gap locks can co-exist. A gap lock taken by one transaction does not prevent another transaction from taking a gap lock on the same gap. There is no difference between shared and exclusive gap locks. They do not conflict with each other, and they perform the same function.)

  Next-Key Lock: Record Lock + Gap Lock,锁定一个范围,包括记录本身。可以解决Phantom Problem,在默认的事务隔离级别下,REPEATABLE READ 采用该锁技术。

2. Next-Key Lock 的锁机制

(1)有索引情况

  当索引具有唯一属性时

    ① 查具体行时,Next-Key Lock会降级为Record Lock,即仅锁住索引本身,而不是范围;

    ② 查小于某值时,Next-Key Lock会锁定小于该值的所有行,并锁定到下一个键值的范围;

    ③ 查大于某值时,Next-Key Lock会锁定大于该值的所有行,并锁定到前一个键值的范围;

  当索引列为辅助索引,Next-Key Lock会为聚集索引和辅助索引分别加锁,聚集索引相等时在该索引上只加Record Lock,辅助索引正常加,聚集索引也是范围查找时,应该会加GAP锁。

  两种显式关闭Next-Key Lock的方式:事务隔离级别改为READ COMMITTED;设置innodb_locks_unsafe_for_binlog为1

(2)没有索引情况

  因为没有索引键值的时候,自动隐式创建索引会锁定整个区间。但依然是行锁而不是表锁,只是等价于表锁。

锁的问题

1. 脏读

  脏读,是事务读取到脏数据。脏数据是事务对缓冲池中行记录的修改,还没有被提交。如READ UNCOMMIT事务隔离级别,违反了事务隔离性

  脏页,是指事务提交后在缓冲池中已被修改的页,但还没有刷新到磁盘中,即数据库实例内存中的页与磁盘中的页数据不一致,当然当数据刷新到磁盘之前,日志都已经被写到重做日志文件中。

  对脏页的读取是正常的,因为脏页的刷新是异步的,内存中和磁盘最终会达到一致性。这种方式不影响数据的可用性,还可以带来性能的提高。

2. 不可重复读(幻读)

  在一个事务内两次查询得到的数据不一致的情况,为不可重复读。如READ COMMITTED事务隔离级别,违反事务一致性

  在默认的事务隔离级别下,REPEATABLE READ 采用Next-Key Lock锁技术,锁定索引范围,可避免该现象。

3. 丢失更新

  在READ UNCOMMIT事务隔离级别下,并发事务可以相互覆盖其他事务的更改,导致丢失更新。

死锁

  死锁是只两个或两个以上的事务在执行过程中,因争夺资源而造成的一种互相等待的现象

  解决死锁方式:

    ① 最简单的方式是超时,当事务等待时间超过时间阈值,进行回滚。根据FIFO的顺序选择回滚对象时,若超时的事务所占权重比较大,即更新操作很多行,占用较多的undo log,回滚该事务比回滚其他事务占用时间长,该方式就不合适。

    ② 等待图(wait-for graph),一种主动的死锁检测方式,要求数据库存储两种信息:锁的信息链表和事务的等待链表,前者记录当前事务加锁信息,后者记录事务的等待情况。

    通过这两张表可以构建一张事务等待图,当出现回路,表示存在死锁。此时,InnoDB选择回滚undo量最小的事务。

锁升级

  InnoDB存储引擎不存在锁升级的问题。

  其不是根据每个记录来产生行锁的,而是根据每个事务访问的每个页对锁进行管理的(通过意向锁为行记录加锁),故事务锁住页中一个记录或多个记录,锁开销是一样的。

  对锁的管理是采用位图的方式(待研究),假如一张表有3 000 000(3M)个数据页,每页大约100条记录,共300 000 000条记录,若、假如一个事务对全表加X锁,若行锁根据每行记录进行锁定,每个锁占用10字节,则锁的管理就需要差不多3G内存。而InnoDB根据页加锁,采用位图的方式,假如每个页存储的锁信息占用30个字节,则所对象仅需30M的内存。

参考

《MySQL技术内幕-InnoDB存储引擎》

详细介绍MySQL/MariaDB的锁

MySQL-InnoDB锁(一)的更多相关文章

  1. mysql: 关于MySQL InnoDB锁行还是锁表?

          baidu zone - 关于MYSQL Innodb 锁行还是锁表,深入讲解

  2. mysql innodb锁简析(2)

    继续昨天的innodb锁的分析: 注:此博文参考一下地址,那里讲的也很详细.http://xm-king.iteye.com/blog/770721 mysql事务的隔离级别分为四种,隔离级别越高,数 ...

  3. [转]关于MYSQL Innodb 锁行还是锁表

    关于mysql的锁行还是锁表,这个问题,今天算是有了一点头绪,mysql 中 innodb是锁行的,但是项目中居然出现了死锁,锁表的情况.为什么呢?先看一下这篇文章. 目时由于业务逻辑的需要,必须对数 ...

  4. MySQL InnoDB锁机制

    概述: 锁机制在程序中是最常用的机制之一,当一个程序需要多线程并行访问同一资源时,为了避免一致性问题,通常采用锁机制来处理.在数据库的操作中也有相同的问题,当两个线程同时对一条数据进行操作,为了保证数 ...

  5. mysql InnoDB锁等待的查看及分析

    说明:前面已经了解了InnoDB关于在出现锁等待的时候,会根据参数innodb_lock_wait_timeout的配置,判断是否需要进行timeout的操作,本文档介绍在出现锁等待时候的查看及分析处 ...

  6. RDS MySQL InnoDB 锁等待和锁等待超时的处理

    https://help.aliyun.com/knowledge_detail/41705.html 1. Innodb 引擎表行锁等待和等待超时发生的场景 2.Innodb 引擎行锁等待情况的处理 ...

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

    MySQL InnoDB支持三种行锁定方式: l   行锁(Record Lock):锁直接加在索引记录上面,锁住的是key. l   间隙锁(Gap Lock):锁定索引记录间隙,确保索引记录的间隙 ...

  8. 你需要知道的MySQL&InnoDB锁都在这里

    目录 一.前言 二.锁的类型 2.1 全局锁 2.2 表级锁 2.2.1 表锁 2.2.2 元数据锁(Meta Data Locks) 2.2.3 自增列锁(AUTO-INC Locks) 2.2.4 ...

  9. mysql——InnoDB 锁

    https://www.cnblogs.com/leedaily/p/8378779.html 1.InnoDB锁的实现方式:给索引项加锁,只有通过索引条件检索数据,InnoDB才使用行级锁,否则,I ...

  10. Mysql Innodb 锁机制

    latch与lock latch 可以认为是应用程序中的锁,可以称为闩锁(轻量级的锁) 因为其要求锁定的时间必须要非常短,若持续时间长,则会导致应用性能非常差,在InnoDB存储引擎中,latch又可 ...

随机推荐

  1. IDEA连接 Oracle数据库

    package com.zxx.util;import org.apache.commons.dbutils.DbUtils;import java.sql.Connection;import jav ...

  2. ZooKeeper的安装及部署

    Zookeeper的安装部署 2.1 Zookeeper的安装 Zookeeper安装前需要安装好 JDK.配置好环境变量. 下载:zookeeper-3.4.5-cdh5.7.0.tar.gz 解压 ...

  3. Android的视图(View)组件

    Android的绝大部分UI组件都放在android.widget包及其子包.android,view包及其子包中,Android应用的所有UI组件都继承了View类,View组件非常类似于Swing ...

  4. 倍增-RMQ问题Sparse-Table算法

       提示 code: #include<bits/stdc++.h> #define ll long long #define inf 0x7fffffff using namespac ...

  5. Java 日志系统

    Java 日志系统 1. 创建日志记录器 private final Logger logger = LoggerFactory.getLogger(LoggerTest.class); 2. 打印日 ...

  6. 10.Bash的安装

    10.Bash的安装本节提供了在 Bash支持的不同系统上的基本安装指导.本版本支持 GNU操作系统,几乎每个 UNIX版本,以及几个非 UNIX 系统,例如 BeOS 和 Interix.还有针对 ...

  7. java——包装类中的equals方法

    基本数据类型包装类中的equals方法用于比对相同包装类中的值是否相等,如果两者比较的包装类类型不同则返回false: Byte public boolean equals(Object obj) { ...

  8. DPI,像素,英寸的关系

    DPI * 英寸 = 像素 eg:在150dpi打印 (2 x 4)英寸的照片 等到多少像素的图像 (150*2)x (150*4) =300 x 600像素 同理可得  已知像素.英寸大小 求DPI ...

  9. linux系统编程相关

    基本的概念:程序,进程,并发,单道程序设计,多道程序设计,时钟中断. 存储介质:寄存器(操作系统的位数是针对寄存器而言的,32位识字节,64位就是8字节).缓存cache.内存,硬盘,网络. cpu的 ...

  10. django 上传路径至vue处理组件加载

    1,在主目录(项目目录)下新建中间件middleware.py文件 写入 from django.utils.deprecation import MiddlewareMixin from djang ...