解决超卖问题,常见的方式,利用redis 的原子性去递减;利用队列,队列入队计数。或者直接打到mysql 层。由mysql 保证不超卖,有几个玩法。利用属性不一样,挺有意思,记录下。

首先,mysql 隔离级别是RR,或者是串行,但是不可能用串行,太慢。

其次,为什么会出现超卖问题?因为这个select and update 是个非原子操作,是两步操作。

最后,怎么解决这个问题? 就在原子性上做文章,比如redis 的 lua 封装,将指令封装成原子操作。或者mysql 的互斥锁,再或者,干脆允许超卖,业务层做二次检查。

第一种,mysql 在select 的时候不加互斥锁,这个时候的做法:

1:开启事务

2:查询库存,并显示的设置写锁(排他锁):SELECT * FROM table_name WHERE …

3:生成订单

4:去库存,隐示的设置写锁(排他锁):UPDATE goods SET counts = counts – 1 WHERE id = 1

5:commit,释放锁

这里提下mysql 的RC 和RR 区别,mysql 是如何实现可重复读的? RR隔离级别是在事务开始时刻,确切地说是第一个读操作创建read view的;RC隔离级别是在语句开始时刻创建read view的。一个read view 可以理解成一份快照(底层只是事务id 列表),所以,RC 隔离级别下,多次读可能受其他事务commit 导致读取数据不一致。

那么 RR 级别下,select 和 update 的时候, 数据是一个version的,但是上面做法的问题,就在于,几个事务begin 后,都读到数据为1,就开始 update ,就会导致数据最终为负数,这种方式,避不了为负数。

我们可以做个二次容错,检查update 后数据为负数后,直接回滚这次事务即可。

第二种方式是select 的时候就加行锁,select for update ,直接锁住这条记录,不让其他事务读。所以对这个数据,都成串行,这个缺点就是会影响性能,但是不会出现超卖的情况了。

1:开启事务

2:查询库存,并显示的设置写锁(排他锁):SELECT * FROM table_name WHERE … FOR UPDATE

3:生成订单

4:去库存,隐示的设置写锁(排他锁):UPDATE goods SET counts = counts – 1 WHERE id = 1

5:commit,释放锁。

第三种,试着给select 加读锁,这种做法是不行,会出现死锁,什么情况会死锁呢?事务1,2同时加读锁,事务1加写锁等事务2的写锁,事务2的加写锁等事务1的读锁,相互等待,陷入死锁。那为啥 select for update 不会这样,因为mysql 互斥锁是可重入锁啊。

mysql 超卖_mysql 解决超卖问题的锁分析相关推荐

  1. mysql 主从 问题_Mysql解决主从不同步问题

    1. 该方法适用于主从库数据相差不大,或者要求数据可以不完全统一的情况,数据要求不严格的情况 stop slave; set global sql_slave_skip_counter =1;#表示跳 ...

  2. mysql varbinary 乱码_mysql解决中文乱码

    mysql>use mydb; mysql>alter database mydb  character set utf8;! 这种方法只对设置后重新创建的表有效,对已存在的表无效 des ...

  3. mysql 特殊字符支持_mysql 解决生僻字,特殊字符插入失败

    MySQL 的 utf8 实际上不是真正的 UTF-8.utf8 只支持每个字符最多三个字节,而真正的 UTF-8 是每个字符最多四个字节.MySQL 一直没有修复这个 bug,他们在 2010 年发 ...

  4. mysql 随机函数 效率_MySQL 随机函数获取数据速度和效率分析

    在mysql中带了随机取数据的函数,在mysql中我们会有rand()函数,很多朋友都会直接使用,如果几百条数据肯定没事,如果几万或百万时你会发现,直接使用是错误的.下面我来介绍随机取数据一些优化方法 ...

  5. mysql xid原理_MySQL数据库分布式事务XA实现原理分析

    [IT168 技术]MySQL XA原理 MySQL XA分为两类,内部XA与外部XA;内部XA用于同一实例下跨多个引擎的事务,由大家熟悉的Binlog作为协调者;外部XA用于跨多MySQL实例的分布 ...

  6. mysql 表死锁_MySQL Innodb表导致死锁日志情况分析与归纳

    案例描述在定时脚本运行过程中,发现当备份表格的sql语句与删除该表部分数据的sql语句同时运行时,mysql会检测出死锁,并打印出日志. 两个sql语句如下:(1)insert into backup ...

  7. MySQL mdl导入_MySQL源码学习——MDL字典锁

    什么是MDL MDL,Meta Data lock,元数据锁,一般称为字典锁.字典锁与数据锁相对应.字典锁是为了保护数据对象被改变,一般是一些DDL会对字典对象改变,如两个TX,TX1先查询表,然后T ...

  8. mysql数据类型设计说明_MySQL的数据类型和建库策略分析详解

    MySQL的数据类型和建库策略分析详解 更新时间:2008年04月06日 01:16:47   作者: 无论是在小得可怜的免费数据库空间或是大型电子商务网站,合理的设计表结构.充分利用空间是十分必要的 ...

  9. redis mysql 解决超卖_Redis 分布式锁解决超卖问题

    Redis 分布式锁解决超卖问题 1,Redis 事物介绍 1. Redis 事物是可以一次执行多个命令, 本质是一组命令的集合. 2. 一个事务中的所有命令都会序列化, 按顺序串行化的执行而不会被其 ...

最新文章

  1. php设置低于设定值不能用,php memory limit怎么设置不限制
  2. argmax最经典解释
  3. mysql查询操作及正则表达式小结
  4. 【物联网】WiFi基础知识
  5. RocketMQ消费者是如何获取消息的?转疯了!
  6. 开源虚拟示波器-_一个新的开源数据库,TP-Link路由器上的开源固件以及更多新闻
  7. 剑指Offer——和为s的两个数字
  8. mysql 主从 均衡_Mysql主从复制
  9. JMeter:生成漂亮的多维度的HTML报告
  10. 《Spring Recipes》第二章笔记:Customizing Bean Initiali...
  11. Android应用程序组件间通信(二)——IntentFilter类简介
  12. 国内maven镜像,快的飞起
  13. 1283 最简单的计算机
  14. con和com开头单词规律_日语记忆其实很有规律,对于日语初学者你知道这些窍门么...
  15. html可以转换wps嘛,如何将图文并茂的网页快速转换为WPS或word文档
  16. 【CV】细粒度图像分割 (FGIS)
  17. round()函数的使用方法
  18. 一台电脑已经连接WIFI,另一台电脑想通过网线连接到网络
  19. MapReduce Inverted Index
  20. Pytorch不同层设置不同学习率

热门文章

  1. C51模拟PS2键盘(二)
  2. js中如何将字符串转化为时间,并计算时间差
  3. Ubuntu查看一些版本 1
  4. 无线通信基础知识12:数字通信之信源编码
  5. python视频教程全集-Python视频教程全集带你入门
  6. 模仿360安全卫士项目笔记8
  7. 杂交水稻 CET4/6
  8. 学习笔记—增量式PID详细实现(C语言)
  9. 【SVM支持向量机】实现和案例
  10. 【图像复原】RDN论文详解(Residual Dense Network for Image Restoration)