关于mysql: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`)
) ENGINE=InnoDB

插入数据:

INSERT INTO `goods` VALUES ('1', 'iphone', '1000');

A、B两个用户同时购买一件id=1的商品:

  • t1: 用户A查问到库存量=1000;
  • t2: 用户B查问到库存量=1000;
  • t3: 用户A下单,将库存量批改为999;
  • t4: 用户B下单,将库存量批改为999;

此时,库存量的数据产生不统一问题。

解决办法:乐观锁和乐观锁两种计划。

乐观锁计划

查问商品时,通过select…for update对该商品记录加锁,其它用户将阻塞期待拜访该记录:

begin
select * from goods where id = 1 for update;        //加锁
update goods set stock = stock - 1 where id = 1;    //下单
commit

因为加锁阻塞其它用户的拜访,对并发拜访不敌对。

乐观锁计划(举荐)

查问商品时不加锁,在更新时,比拟mem.cur_val=db.val:

  • 若相等,则间接用cu_val更新db;
  • 否则,从新获取mem.cur_val,再次比拟mem.cur_val=db.val;
select stock from goods where id = 1;    //失去mem.cur_val

begin
//仅当stock=cur_val时才更新
update goods set stock = stock - 1 where id = 1 and stock = cur_val;
commit

乐观锁因为不阻塞其它用户的拜访,并发度较高,适宜读取频繁的场景。

参考

1.https://mp.weixin.qq.com/s/8u…

评论

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

这个站点使用 Akismet 来减少垃圾评论。了解你的评论数据如何被处理