参考 https://www.cnblogs.com/yaochunhui/p/14186371.html
https://www.cnblogs.com/tutar/p/5878651.html
锁的作用
Mysql中锁主要用来保障线程在并发读写数据的时候数据的安全性的。他其实跟操作系统中的锁本质差不多, 只是增加了业务特性。操作系统中
锁的种类
锁细分的话可以分为 :
共享锁 : 其他事务可以读,但不能写。
排它锁 : 其他事务不能读,也不能写。
意向共享锁 : 事务在给数据加共享锁之前必须先获取数据的意向共享锁。
意向排它锁 : 事务在给数据加排它锁之前必须先获取数据的意向排它锁。
自增锁 : 如果表存在自增字段, 则新增数据的时候会使用自增锁锁表(可配置)。
innodb_autoinc_lock_mode:可以设定3个值,0,1,2
0:traditonal (每次都会产生表锁)
1:consecutive (会产生一个轻量锁,simple insert会获得批量的锁,保证连续插入)
2:interleaved (不会锁表,来一个处理一个,并发最高)
间隙锁 : 锁住B+数叶子结点链表中的某一段数据的锁(范围操作时用),锁多行。
记录锁 : 也叫行锁, 锁某条记录。
临键锁 : 记录锁与间隙锁的组合,它的封锁范围,既包含索引记录,又包含索引区间。
关于锁行还是锁表:
Myisam存储引擎只能锁表, 所以对于高并发写的场景性能较差。
Innodb存储引擎只有通过索引列加的锁才是行锁, 否则也会锁表, 间隙锁和记录锁是建立在索引的基础上的。
表锁: 开销小,加锁快;不会出现死锁;锁定力度大,发生锁冲突概率高,并发度最低。
行锁: 开销大,加锁慢;会出现死锁;锁定粒度小,发生锁冲突的概率低,并发度高 不同的存储引擎支持的锁粒度是不一样的。
InnoDB行锁和表锁都支持、MyISAM只支持表锁!
InnoDB只有通过索引条件检索数据才使用行级锁,否则,InnoDB使用表锁也就是说,InnoDB的行锁是基于索引的!
两阶段提交 :
在 InnoDB 事务中,行锁是在需要的时候才加上的,但并不是不需要了就立刻释放,而是要等到事务结束时才释放。这个就是两阶段锁协议
死锁 :
当并发系统中不同线程出现循环资源依赖,涉及的线程都在等待别的线程释放资源时,就会导致这几个线程都进入无限等待的状态,称为死锁。
如下例子
这时候,事务 A 在等待事务 B 释放 id=2 的行锁,而事务 B 在等待事务 A 释放 id=1 的行锁。 事务 A 和事务 B 在互相等待对方的资源释放,就是进入了死锁状态。
当出现死锁以后,有两种策略:
一种策略是,直接进入等待,直到超时。这个超时时间可以通过参数 innodb_lock_wait_timeout 来设置。
另一种策略是,发起死锁检测,发现死锁后,主动回滚死锁链条中的某一个事务,让其他事务得以继续执行。将参数 innodb_deadlock_detect 设置为 on,表示开启这个逻辑。
锁的具体使用
- mysql对于事务默认开启了自动提交
- 除了显式的声明 begin 这种,其实每条sql语句都是一个事务
- innodb中的加锁解锁机制遵循MVCC里面的原则 :
快照读:简单的select操作,属于快照读,不加锁。(当然,也有例外,下面会分析)
select * from table where ?;
当前读:特殊的读操作,插入/更新/删除操作,属于当前读,需要加锁。
select * from table where ? lock in share mode;
select * from table where ? for update;
insert into table values (…);
update table set ? where ?;
delete from table where ?;
所有以上的语句,都属于当前读,读取记录的最新版本。并且,读取之后,还需要保证其他并发事务不能修改当前记录,对读取记录加锁。其中,除了第一条语句,对读取记录加S锁 (共享锁)外,其他的操作,都加的是X锁 (排它锁)。
为什么将 插入/更新/删除 操作,都归为当前读?可以看看下面这个 更新 操作,在数据库中的执行流程:
从图中,可以看到,一个Update操作的具体流程。当Update SQL被发给MySQL后,MySQL Server会根据where条件,读取第一条满足条件的记录,然后InnoDB引擎会将第一条记录返回,并加锁 (current read)。待MySQL Server收到这条加锁的记录之后,会再发起一个Update请求,更新这条记录。一条记录操作完成,再读取下一条记录,直至没有满足条件的记录为止。因此,Update操作内部,就包含了一个当前读。同理,Delete操作也一样。Insert操作会稍微有些不同,简单来说,就是Insert操作可能会触发Unique Key的冲突检查,也会进行一个当前读。
注:根据上图的交互,针对一条当前读的SQL语句,InnoDB与MySQL Server的交互,是一条一条进行的,因此,加锁也是一条一条进行的。先对一条满足条件的记录加锁,返回给MySQL Server,做一些DML操作;然后在读取下一条加锁,直至读取完毕。
- 在innodb中加行锁是根据你命中的索引来的, 如果你没有命中索引, 则会加表锁。
- Myisam同一时刻只能执行一条update语句, 因为他会锁表, 所以他的update并发很低。但是如果是读取的话, myisam锁表只要加锁解锁一次,而innodb可能对多行数据加锁解锁, 所以反而慢一些。