`

(转)如何正解决库存超卖问题 --乱

 
阅读更多

  一般电子商务网站都会遇到如团购、秒杀、特价之类的活动,而这样的活动有一个共同的特点就是访问量激增、上千甚至上万人抢购一个商品。然而,作为活动商品,库存肯定是很有限的,如何控制库存不让出现超买,以防止造成不必要的损失是众多电子商务网站程序员头疼的问题,这同时也是最基本的问题。
  本文来源:http://blog.csdn.net/caomiao2006/article/details/38568825,并就作者相关看法作了少量修改。

[内容]
  注意!文中说到缓存用memcache,并以此作库存操作控制!作者认为应使用redis替代memcache,原因是redis支持事务,原子性与隔离性比memcache好。因此需要注意此点。

  从技术方面剖析,很多人肯定会想到事务,但是事务是控制库存超卖的必要条件,但不是充分必要条件。
举例:总库存:4个商品,请求人:a、1个商品 b、2个商品 c、3个商品
示例如下:

begin...
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...

  以上代码就是我们平时控制库存写的代码了,大多数人都会这么写,看似问题不大,其实隐藏着巨大的漏洞。

  • 数据库的访问其实就是对磁盘文件的访问,数据库中的表其实就是保存在磁盘上的一个个文件,甚至一个文件包含了多张表。例如由于高并发,当前有三个用户a、b、c三个用户进入到了这个事务中,这个时候会产生一个共享锁,所以在select的时候,这三个用户查到的库存数量都是4个。
  • 同时还要注意,mysql innodb查到的结果是有版本控制的,再其他用户更新没有commit之前(也就是没有产生新版本之前),当前用户查到的结果依然是旧版本;
  • 接着执行update,假如这三个用户同时到达update这里,这个时候update更新语句会把并发串行化,也就是给同时到达这里的是三个用户排个序,一个一个执行,并生成排他锁,在当前这个update语句commit之前,其他用户等待执行,commit后,生成新的版本;这样执行完后,库存肯定为负数了。

  下面我们来改造一下代码:

begin....
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){
       thrownewException('库存不足');
    }
}catch($e Exception){
    rollBack(回滚)
}
commit....

  再优化一下,如:

begin...
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,这样其他线程将无法提交数据。


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

[结论]
1,不建议在数据库层面加锁,建议通过服务端的内存锁(锁主键)。当某个用户要修改某个id的数据时,把要修改的id存入memcache,若其他用户触发修改此id的数据时,读到memcache有这个id的值时,就阻止那个用户修改。

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

3,注意防止缓存穿透问题。即:如果缓存中所存商品的id范围可知,如100-1000,那么当我查询id为<100及1000的商品时,这里缓存中肯定是不存在的,按逻辑肯定会去查DB,查到后把数据放到缓存。这时问题来了,如果有人故意刷你这个范围外的东西,这样每次去查DB,并发量上来时,DB一定会挂。因此,为了避免这个问题,应在缓存放置当前数据存在的记录ID,通过ID确定是否需要查询DB。这样大大减轻了DB的压力。

 

 

REFS:http://blog.163.com/sujoe_2006/blog/static/335315120141139728398/

 

分享到:
评论

相关推荐

    Spring Boot + redis解决商品秒杀库存超卖

    在众多抢购活动中,在有限的商品数量的限制下如何保证抢购到商品的用户数不能大于商品数量,也就是不能出现超卖的问题;还有就是抢购时会出现大量用户的访问,如何提高用户体验效果也是一个问题,也就是要解决秒杀...

    使用rabbitmq解决超卖问题

    为了解决这个问题,我们可以利用消息队列,特别是RabbitMQ这样的中间件来实现分布式事务和异步处理,确保库存的准确性和系统稳定性。本示例将详细介绍如何使用RabbitMQ来避免超卖问题。 首先,RabbitMQ是一个开源的...

    Redis中的String类型及使用Redis解决订单秒杀超卖问题

    本系列将和大家分享Redis分布式缓存,本章主要简单介绍下Redis中的String类型,以及如何使用Redis解决订单秒杀超卖问题。 Redis中5种数据结构之String类型:key-value的缓存,支持过期,value不超过512M。 Redis是...

    解决高并发环境下Redis连接超时与超卖问题

    这样,即使在高并发情况下,如果多个用户同时尝试购买同一商品,只有一个用户的请求能够成功完成交易,其余用户则会在检查库存时发现库存已减少,从而避免了超卖问题。 同时,为了应对连接超时,我们需要优化Redis...

    PHP+Redis链表解决高并发下商品超卖问题(实现原理及步骤)

    上一篇文章聊了一下使用Redis事务来解决高并发商品超卖问题,今天我们来聊一下使用Redis链表来解决高并发商品超卖问题。 实现原理 使用redis链表来做,因为pop操作是原子的,即使有很多用户同时到达,也是依次执行,...

    Redis从入门到精通(七)Redis实战(四)库存超卖与一人一单 测试项目代码

    在本节中,我们将深入探讨Redis在实际应用中的库存超卖和一人一单问题,并通过提供的测试项目代码“redis_learning_dzdp”进行实践。Redis作为一个高性能的键值存储系统,常常被用作分布式环境下的解决方案,特别是...

    微服务 Spring Boot 整合Redis 秒杀 ,全局唯一ID,乐观锁解决库存超卖,Jmeter 测试 每秒千万级并发

    本项目基于微服务架构,使用Spring Boot整合Redis,实现了高效的秒杀系统,并通过全局唯一ID(UUID)和乐观锁技术来防止库存超卖问题。此外,还利用JMeter进行性能测试,确保系统能在每秒处理千万级别的并发请求。...

    通达信指标公式源码 超买超卖-超级KDJ.doc

    通达信指标公式源码 超买超卖-超级KDJ.doc 本资源是一个通达信指标公式源码,名为超买超卖-超级KDJ.doc。该指标是在传统 KDJ 指标基础上对其做了 MA 均线的平滑处理,避免信号过多和繁杂。 指标介绍 该指标的主要...

    超卖问题的4种解决方案、秒杀方案

    超卖问题的4种解决方案,乐观锁、逻辑控制方式来解决超卖问题,都是利用数据库来实现 方案1:通过update中携带条件判断解决超卖问题 方案2:乐观锁解决超卖问题 方案3:对比数据修改前后是否和期望的一致,解决超卖...

    一个微服务 Spring Boot 整合Redis 秒杀 ,全局唯一ID,乐观锁解决库存超卖,Jmeter 测试 每秒千万级并发

    同时,为了防止库存超卖,我们将采用乐观锁策略,并通过JMeter进行性能测试,确保系统能够处理每秒千万级别的并发请求。以下是关于这些技术点的详细解释: 1. **Spring Boot 微服务**: Spring Boot简化了Spring...

    ECSHOP多城市多仓库多库存插件 UTF-8版

    "ECSHOP多城市多仓库多库存插件 UTF-8版" 是针对ECSHOP系统的一个扩展功能,旨在解决商家在运营过程中可能遇到的复杂库存管理问题。这个插件的核心特点是支持多个城市的仓库管理和库存同步,使得商家能够更有效地...

    Redis的类库及Redis解决订单秒杀超卖问题,极光推送代码

    在本文中,我们将重点讨论Redis的类库、String类型的应用,以及如何利用Redis来解决订单秒杀超卖的问题。同时,还会涉及C# .Net Core下极光推送的代码实现。 首先,让我们来看看C# .Net Core下如何使用Redis的类库...

    seckill-master-lkk.zip

    - 秒杀的商品一般库存较少,只有少数用户能够购买,要控制好库存,防止超卖; - 整个系统关键在于支撑短时间内的高并发,降低数据库压力,业务和普通商品购买区别不大 秒杀系统性能瓶颈在于数据库无法处理并发...

    库存管理系统 -ASP源码.zip

    4. **订单管理**:处理客户订单,确保库存足够,防止超卖,并记录订单详情。 5. **报告和统计**:生成库存报表,展示库存周转率、滞销品等关键指标,辅助决策。 文件列表中的"内容来自存起来软件站...

    高并发场景防止库存数量超卖少卖

    在高并发电商场景下,商品超卖(即销售量超出库存)是常见问题,主要由并发扣减库存导致。常规做法是在扣减库存前检查库存充足性,但面对大量并发请求时,这种方法可能失效。为此,可采用以下几种策略: 悲观锁:在...

    高性能电商秒杀解决方案

    • 秒杀的商品一般库存较少,只有少数用户能够购买,要控制好库存,防止超卖; • 整个系统关键在于支撑短时间内的高并发,降低数据库压力,业务和普通商品购买区别不大 秒杀系统性能瓶颈在于数据库无法处理并发访问...

    超卖和分布式锁解决方案.docx

    综上所述,通过以上各种方法和技术的应用,可以有效地解决在高并发场景下超卖问题以及分布式环境下的资源同步问题。这些解决方案不仅可以提高系统的稳定性,还能优化用户体验,对于构建高性能的分布式系统具有重要的...

    用友培训课件:U8 网络分销12_1-库存业务.pptx

    综上,用友U8网络分销12.1的库存业务涵盖广泛,从基础的库存操作到复杂的库存控制策略,为企业提供了全面的库存管理解决方案。通过这些功能,企业能够有效监控库存,避免库存积压或短缺,优化供应链管理,提高运营...

    ECSHOP多地区仓库库存插件开源版-共享插件

    这款多地区仓库库存插件是针对ECSHOP系统设计的,旨在解决商家在不同地区设有仓库,需要管理多个仓库库存的问题。开源版本意味着源代码公开,允许用户自由查看、修改和分发,这对初学者来说是一个很好的学习和实践...

Global site tag (gtag.js) - Google Analytics