mysql锁

参考 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,表示开启这个逻辑。

锁的具体使用

  1. mysql对于事务默认开启了自动提交
  2. 除了显式的声明 begin 这种,其实每条sql语句都是一个事务
  3. 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操作;然后在读取下一条加锁,直至读取完毕。

  1. 在innodb中加行锁是根据你命中的索引来的, 如果你没有命中索引, 则会加表锁。
  2. Myisam同一时刻只能执行一条update语句, 因为他会锁表, 所以他的update并发很低。但是如果是读取的话, myisam锁表只要加锁解锁一次,而innodb可能对多行数据加锁解锁, 所以反而慢一些。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值