mysql处理高并发,防止库存超卖[亲测有效]

mysql处理高并发,防止库存超卖[亲测有效]一般电子商务网站都会遇到如团购、秒杀、特价之类的活动,而这样的活动有一个共同的特点就是访问量激增、上千甚至上万人抢购一个商品。然而,作为活动商品,库存肯定是很有限的,如何控制库存不让出现超买,以防止…

mysql处理高并发,防止库存超卖

mysql处理高并发,防止库存超卖[亲测有效]

一般电子商务网站都会遇到如团购、秒杀、特价之类的活动,而这样的活动有一个共同的特点就是访问量激增、上千甚至上万人抢购一个商品。然而,作为活动商品,库存肯定是很有限的,如何控制库存不让出现超买,以防止造成不必要的损失是众多电子商务网站程序员头疼的问题,这同时也是最基本的问题。

从技术方面剖析,很多人肯定会想到事务,但是事务是控制库存超卖的必要条件,但不是充分必要条件。

举例:

总库存:4个商品

请求人:a、1个商品 b、2个商品 c、3个商品

程序如下:

beginTranse(开启事务)

try{

    $result = $dbca->query("select amount from s_store where postID = 12345");

    if(result->amount > 0){

        //quantity为请求减掉的库存数量

        $dbca->query("update s_store set amount = amount - quantity where postID = 12345");

    }

}catch($e Exception){

    rollBack(回滚)

}

commit(提交事务)

代码100分

以上代码就是我们平时控制库存写的代码了,大多数人都会这么写,看似问题不大,其实隐藏着巨大的漏洞。数据库的访问其实就是对磁盘文件的访问,数据库中的表其实就是保存在磁盘上的一个个文件,甚至一个文件包含了多张表。

例如由于高并发,当前有三个用户a、b、c三个用户进入到了这个事务中,这个时候会产生一个共享锁,所以在select的时候,这三个用户查到的库存数量都是4个,同时还要注意,mysql innodb查到的结果是有版本控制的,再其他用户更新没有commit之前(也就是没有产生新版本之前),当前用户查到的结果依然是就版本;

然后是update,假如这三个用户同时到达update这里,这个时候update更新语句会把并发串行化,也就是给同时到达这里的是三个用户排个序,一个一个执行,并生成排他锁,在当前这个update语句commit之前,其他用户等待执行,commit后,生成新的版本;这样执行完后,库存肯定为负数了。但是根据以上描述,我们修改一下代码就不会出现超买现象了,代码如下:

代码100分beginTranse(开启事务)

try{

    //quantity为请求减掉的库存数量
    $dbca->query("update s_store set amount = amount - quantity where postID = 12345");

    $result = $dbca->query("select amount from s_store where postID = 12345");

    if(result->amount < 0){

       throw new Exception("库存不足");

    }

}catch($e Exception){

    rollBack(回滚)

}

commit(提交事务)

另外,更简洁的方法:

beginTranse(开启事务)

try{

    //quantity为请求减掉的库存数量
    $dbca->query("update s_store set amount = amount - quantity where amount>=quantity and postID = 12345");

}catch($e Exception){

    rollBack(回滚)

}

commit(提交事务)

1、在秒杀的情况下,肯定不能如此高频率的去读写数据库,会严重造成性能问题的必须使用缓存,将需要秒杀的商品放入缓存中,并使用锁来处理其并发情况。

当接到用户秒杀提交订单的情况下,先将商品数量递减(加锁/解锁)后再进行其他方面的处理,处理失败在将数据递增1(加锁/解锁),否则表示交易成功。

当商品数量递减到0时,表示商品秒杀完毕,拒绝其他用户的请求。

2、这个肯定不能直接操作数据库的,会挂的。直接读库写库对数据库压力太大,要用缓存。
把你要卖出的商品比如10个商品放到缓存中;然后在memcache里设置一个计数器来记录请求数,这个请求书你可以以你要秒杀卖出的商品数为基数,比如你想卖出10个商品,只允许100个请求进来。

那当计数器达到100的时候,后面进来的就显示秒杀结束,这样可以减轻你的服务器的压力。然后根据这100个请求,先付款的先得后付款的提示商品以秒杀完。

3、首先,多用户并发修改同一条记录时,肯定是后提交的用户将覆盖掉前者提交的结果了这个直接可以使用加锁机制去解决,乐观锁或者悲观锁。

乐观锁,就是在数据库设计一个版本号的字段,每次修改都使其+1,这样在提交时比对提交前的版本号就知道是不是并发提交了,但是有个缺点就是只能是应用中控制,如果有跨应用修改同一条数据乐观锁就没办法了,这个时候可以考虑悲观锁。

悲观锁,就是直接在数据库层面将数据锁死,类似于oralce中使用select xxxxx from xxxx where xx=xx for update,这样其他线程将无法提交数据。

除了加锁的方式也可以使用接收锁定的方式,思路是在数据库中设计一个状态标识位,用户在对数据进行修改前,将状态标识位标识为正在编辑的状态,这样其他用户要编辑此条记录时系统将发现有其他用户正在编辑,则拒绝其编辑的请求,类似于你在操作系统中某文件正在执行,然后你要修改该文件时,系统会提醒你该文件不可编辑或删除。

4、不建议在数据库层面加锁,建议通过服务端的内存锁(锁主键)。

当某个用户要修改某个id的数据时,把要修改的id存入memcache,若其他用户触发修改此id的数据时,读到memcache有这个id的值时,就阻止那个用户修改。

5、实际应用中,并不是让mysql去直面大并发读写,会借助“外力”,比如缓存、利用主从库实现读写分离、分表、使用队列写入等方法来降低并发读写。

 

mysql处理高并发,防止库存超卖[亲测有效]

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
转载请注明出处: https://daima100.com/8220.html

(0)
上一篇 2023-03-05
下一篇 2023-03-05

相关推荐

  • MySQL总结(三)DDL数据定义语言-详解

    MySQL总结(三)DDL数据定义语言-详解1.DDL操作数据库 1.1 创建数据库 1.1.1 多种方式创建数据库 语句 说明 CREATE DATABASE 数据库名称; 创建数据库 CREATE DATABASE IF NOT EXIS…

    2023-03-01
    132
  • nosql介绍_iaas paas saas举例

    nosql介绍_iaas paas saas举例王奇 顾问软件工程师 目前从事 PaaS 中间件服务(Redis / MongoDB / ELK 等)开发工作,对 NoSQL 数据库有深入的研究以及丰富的二次开发经验,热衷对 NoSQL 数据库领域

    2023-05-03
    164
  • 详解数仓的锁相关参数及视图[通俗易懂]

    详解数仓的锁相关参数及视图[通俗易懂]摘要:GaussDB(DWS) 中锁等待可以设置等待超时相关参数,一旦等锁的时间超过参数配置值会抛错。 本文分享自华为云社区《GaussDB(DWS) 锁相关参数及视图详解》,作者: yd_22052

    2023-06-09
    129
  • 解决MySql客户端秒退(找不到my.ini)[通俗易懂]

    解决MySql客户端秒退(找不到my.ini)[通俗易懂]问题说明(环境:windows7,MySql8.0) 今天安装好MySql后启动MySql服务 启动服务都失败的就不要往下看了,自行百度解决。 打开客户端秒退,但在cmd中是可以使用数据库的。 正常来

    2023-01-27
    158
  • MyBatis中动态sql实现传递多个参数并使用if进行参数的判断和实现like模糊搜索以及foreach实现in集合[通俗易懂]

    MyBatis中动态sql实现传递多个参数并使用if进行参数的判断和实现like模糊搜索以及foreach实现in集合[通俗易懂] 场景 在mapper接口层方法参数有id的数组和姓名的模糊搜索和类型的三个参数。 现在需要将这三个参数传递在动态sql的xml中接受并进行判断不为空。 然后查询数据库中记录在第一个参数id数…

    2023-03-25
    155
  • Python如何清空列表

    Python如何清空列表
    列表是Python中非常常用的一种数据类型,它可以容纳多个元素,并且可以通过索引访问到其中的每个元素。
    在Python中,我们经常会遇到需要清空列表数据的情况,比如说在重新使用列表之前,需要将之前的数据清空掉。
    本文将介绍Python中常用的几种清空列表的方法,并且对它们进行详细的讲解。

    2024-05-12
    91
  • Python工程师

    Python工程师随着信息技术的快速发展,越来越多企业开始注重数据分析能力。Python作为一种流行的编程语言,以其简洁、易读、易学、高效的特性在企业中越来越受欢迎,进而催生了一些新兴职业,如Python工程师。Python工程师也成为了近几年IT行业中一种炙手可热的职业。

    2024-04-26
    71
  • Python模块导入:优化模块重载和避免命名冲突

    Python模块导入:优化模块重载和避免命名冲突在Python中,模块是指一个包含了变量、函数、类等语句的.py文件。而模块导入,则是将这些语句引入到当前的程序中,以便使用其中的函数、变量或者类。

    2024-02-21
    197

发表回复

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