MySQL 排它锁 详解

Mysql InnoDB 排他锁
用法 : select … for update;
例如 : select * from goods where id = 1 for update;
排他锁的申请前提 : 没有线程对该结果集中的任何行数据使用排他锁或共享锁,否则申请会阻塞
for update 仅适用于 InnoDB,且必须在事务块 (BEGIN/COMMIT) 中才能生效,在进行事务操作时,通过 for update 语句,MySQL会对查询结果集中每行数据都添加排他锁,其他线程对该记录的更新与删除操作都会阻塞,排他锁包含 行锁、表锁

创建一张商品表 goods,它包含 id,商品名称,库存量三个字段
CREATE TABLE goods(
id INT(11) NOT NULL AUTO_INCREMENT,
name VARCHAR(100) DEFAULT NULL,
stock INT(11) DEFAULT NULL,
PRIMARY KEY (id),
UNIQUE KEY idx_name (name) USING HASH
) ENGINE=InnoDB;

INSERT INTO goods VALUES
('1', 'prod11', '1000'),
('2', 'prod12', '1000'),
('3', 'prod13', '1000'),
('4', 'prod14', '1000'),
('5', 'prod15', '1000'),
('6', 'prod16', '1000'),
('7', 'prod17', '1000'),
('8', 'prod18', '1000'),
('9', 'prod19', '1000');

1> 数据一致性
假设有 A、B 两个用户同时各购买一件 id=1 的商品,用户A获取到的库存量为 1000,用户B获取到的库存量也为 1000,用户A完成购买后修改该商品的库存量为 999,用户B完成购买后修改该商品的库存量为 999,此时库存量数据产生了不一致
有两种解决方案 :
方案1> 悲观锁方案 : 每次获取商品时,对该商品加排他锁,也就是在用户A获取获取 id=1 的商品信息时对该行记录加锁,期间其他用户阻塞等待访问该记录,悲观锁适合写入频繁的场景
BEGIN;
SELECT * FROM goods WHERE id=1 FOR UPDATE;
UPDATE goods SET stock=stock-1 WHERE id = 1;
COMMIT;

方案2> 乐观锁方案 : 每次获取商品时,不对该商品加锁。在更新数据的时候需要比较程序中的库存量与数据库中的库存量是否相等,如果相等则进行更新,反之程序重新获取库存量,再次进行比较,直到两个库存量的数值相等才进行数据更新。乐观锁更适合读取频繁的场景
# 不加锁获取 id=1 的商品对象
SELECT * FROM goods WHERE id=1;

BEGIN;
# 更新 stock 值,这里需要注意 where 条件 "stock=cur_stock",只有程序中获取到的库存量与数据库中的库存量相等才执行更新
UPDATE goods SET stock=stock-1 WHERE id=1 AND stock=cur_stock;
COMMIT;
对于商城系统 查询商品的频率比下单支付的频次高,比较适合乐观锁方案

2> 行锁与表锁
<1> 只根据主键进行查询,并且查询到数据,主键字段产生行锁
BEGIN;
SELECT * FROM goods WHERE id=1 FOR UPDATE;
COMMIT;

<2> 只根据主键进行查询,没有查询到数据,不产生锁
BEGIN;
SELECT * FROM goods WHERE id=10 FOR UPDATE;
COMMIT;

<3> 根据主键、非主键含索引(name) 进行查询,并且查询到数据,主键字段产生行锁,name字段产生行锁
BEGIN;
SELECT * FROM goods WHERE id=1 AND name='prod11' FOR UPDATE;
COMMIT;

<4> 根据主键、非主键含索引(name) 进行查询,没有查询到数据,不产生锁
BEGIN;
SELECT * FROM goods WHERE id=1 AND name='prod12' FOR UPDATE;
COMMIT;

<5> 根据主键、非主键不含索引(stock) 进行查询,并且查询到数据,如果其他线程按主键字段进行再次查询,则主键字段产生行锁,如果其他线程按非主键不含索引字段进行查询,则非主键不含索引字段产生表锁,如果其他线程按非主键含索引字段进行查询,则非主键含索引字段产生行锁,如果索引值是枚举类型,mysql也会进行表锁
BEGIN;
SELECT * FROM goods WHERE id=1 AND stock=2000 FOR UPDATE;
COMMIT;
简单说就是,多个字段查询 线程按照非枚举索引字段查询数据,将会产生锁行,其他情况将会产生锁表

<6> 根据主键、非主键不含索引(stock)查询,没有查询到数据,不产生锁
BEGIN;
SELECT * FROM goods WHERE id=1 AND stock=2000 FOR UPDATE;
COMMIT;

<7> 根据非主键含索引(name) 进行查询,并且查询到数据,name字段产生行锁
BEGIN;
SELECT * FROM goods WHERE name='prod11' FOR UPDATE;
COMMIT;

<8> 根据非主键含索引(name)进行查询,没有查询到数据,不产生锁
BEGIN;
SELECT * FROM goods WHERE name='prod11' FOR UPDATE;
COMMIT;

<9> 根据非主键不含索引(stock)进行查询,并且查询到数据,stock字段产生表锁
BEGIN;
SELECT * FROM goods WHERE stock=2000 FOR UPDATE;
COMMIT;

<10> 根据非主键不含索引(stock)进行查询,没有查询到数据,stock字段产生表锁
BEGIN;
SELECT * FROM goods WHERE stock=2000 FOR UPDATE;
COMMIT;

<11> 只根据主键进行查询,查询条件为不等于,并且查询到数据,主键字段产生表锁
BEGIN;
SELECT * FROM goods WHERE id!=1 FOR UPDATE;
COMMIT;

<12> 只根据主键进行查询,查询条件为不等于,没有查询到数据,主键字段产生表锁
BEGIN;
SELECT * FROM goods WHERE id!=10 FOR UPDATE;
COMMIT;

<13> 只根据主键进行查询,查询条件为 like,并且查询到数据,主键字段产生表锁
BEGIN;
SELECT * FROM goods WHERE id LIKE '1' FOR UPDATE;
COMMIT;

<14> 只根据主键进行查询,查询条件为 like,没有查询到数据,主键字段产生表锁
BEGIN;
SELECT * FROM goods WHERE id LIKE '1' FOR UPDATE;
COMMIT;

1> InnoDB 行锁是通过给索引上的索引项加锁来实现的,只有通过索引条件检索数据,InnoDB 才使用行级锁,否则,InnoDB将使用表锁。
2> 由于 MySQL 行锁是针对索引加的锁,不是针对记录加的锁,所以虽然是访问不同行的记录,但是如果是使用相同的索引键,是会出现锁冲突的,设计的时候要注意这一点
3> 当表有多个索引的时候,不同的事务可以使用不同的索引锁定不同的行,另外,不论是使用主键索引、唯一索引或普通索引,InnoDB都会使用行锁来对数据加锁
4> 即便在条件中使用了索引字段,但是否使用索引来检索数据是由 MySQL 通过判断不同执行计划的代价来决定的,如果 MySQL 认为全表扫描效率更高,比如对一些很小的表,它就不会使用索引,这种情况下 InnoDB 将使用表锁,而不是行锁。因此,在分析锁冲突时,别忘了检查 SQL 的执行计划,以确认是否真正使用了索引
5> 检索值的数据类型与索引字段不同,虽然 MySQL 能够进行数据类型转换,但却不会使用索引,从而导致 InnoDB 使用表锁。通过用explain 检查两条SQL的执行计划


你可能感兴趣的:(MySQL)