广告
返回顶部
首页 > 资讯 > 数据库 >Go+Lua解决Redis秒杀中库存与超卖问题
  • 270
分享到

Go+Lua解决Redis秒杀中库存与超卖问题

GoLuaRedis秒杀GoLuaRedis库存与超卖 2023-03-01 10:03:36 270人浏览 八月长安
摘要

目录0、简介1、简单版2、解决超卖3、解决库存问题lua0、简介 Go语言连接go-Redis进行数据库的连接,如果你对这部分尚不了解,建议你先学习这部分知识。另外,本秒杀主要解决两个问题,第一个就是超卖问题,另一个就是

0、简介

  • Go语言连接go-Redis进行数据库的连接,如果你对这部分尚不了解,建议你先学习这部分知识。
  • 另外,本秒杀主要解决两个问题,第一个就是超卖问题,另一个就是库存问题。
  • 没有设计专门的页面来模拟并发,我们直接使用gorountine,在调用请求前停留10s。
  • 针对超卖问题,引入go-redis的watch搭配事务处理即可【相当于乐观】。

而针对库存的问题较为麻烦一点,需要使用Lua编辑脚本,但是你无需在自己的机器上下载lua的编译环境,go提供了其相关的支持。针对这一部分,不用慌张,其基本架构如下:

Go+Lua解决Redis秒杀中库存与超卖问题

1、简单版

面对并发的情况下会出现超卖的情况,redis数据库中会出现负值的情况。即使你在操作之前进行了数据的判断。

func MsCode(uuid, prodid string) bool { 
 // 1、对uuid和prodid进行非空判断 
 if uuid == "" || prodid == "" { 
   return false 
 } 

 //2、获取连接 
 rdb := DB 

 //3、拼接key 
 kcKey := "kc:" + prodid + ":Qt" 
 userKey := "sk:" + prodid + ":user" 

 //4、获取库存 
 str, err := rdb.Get(ctx, kcKey).Result() 
 if err != nil { 
   fmt.Println(err) 
   fmt.Println("秒杀还未开始.......") 
   return false 
 } 

 // 5、判断用户是否重复秒杀操作 
 flag, err := rdb.SIsMember(ctx, userKey, userKey).Result() 
 if err != nil { 
   fmt.Println(err) 
 } 
 if flag { 
   fmt.Println("你已经参加了秒杀,无法再次参加。。。。") 
   return false 
 } 

 // 6、判断商品数量,如果库存数量小于1,秒杀结束 
 str, err = rdb.Get(ctx, kcKey).Result() 
 if err != nil { 
   fmt.Println(err) 
 } 
 n, err := strconv.Atoi(str) 
 if err != nil { 
   fmt.Println(err) 
 } 
 if n < 1 { 
   fmt.Println("秒杀结束,请下次再来吧。。。。") 
   return false 
 } 

 // 7、秒杀过程 
 // 7.1、库存减1 
 num, err := rdb.Decr(ctx, kcKey).Result() 
 if err != nil { 
   fmt.Println(err) 
 } 
 if num != 0 { 
   // 7.2、添加用户 
   rdb.SAdd(ctx, userKey, uuid) 
 } 
 return true 
}

func main() {
 // 并发的版本
 for i := 0; i < 20; i++ {
  go func() {
   uuid := GenerateUUID()
   prodid := "1023"
   time.Sleep(10 * time.Second)
   MsCode(uuid, prodid)
  }()
 }
 time.Sleep(15 * time.Second)
}

2、解决超卖

使用watch进行监视key,关键部分如下。但是这样会造成一个问题,就是抢购不完,会有一些库存,但是又有人没有抢到。

err = rdb.Watch(ctx, func(tx *redis.Tx) error {  
   n, err := tx.Get(ctx, kcKey).Int()  
   if err != nil && err != redis.Nil {  
      return err  
   }  
   if n <= 0 {  
      return fmt.Errorf("抢购结束了!请下次早点来。。。。")  
   }  
   _, err = tx.TxPipelined(ctx, func(pipeliner redis.Pipeliner) error {  
      err := pipeliner.Decr(ctx, kcKey).Err()  
      if err != nil {  
         return err  
      }  
      err = pipeliner.SAdd(ctx, userKey, uuid).Err()  
      if err != nil {  
         return err  
      }  
      return nil  
   })  
   return err  
}, kcKey)

3、解决库存问题Lua

Lua操作redis能够比较好的解决这个问题。因为redis中使用watch是使用了悲观锁的形态,而悲观锁会自然得造成库存问题,因此要使用乐观锁。而redis天然不支持乐观锁,基于此,需要时lua来编写相关脚本。其主要有以下优势:

  • 将复杂的或者多步的redis操作,写为一个脚本,一次提交给redis执行,减少反复连接redis的次数。提升性能。
  • luan脚本类似redis事务,有一定的原子性,不会被其他命令插队,可以完成一些redis事务性的操作。
  • redis的lua脚本功能,只有在redis2.6以上的版本才可以使用。
  • 利用lua脚本淘汰用户,解决超卖问题。
  • redis2.6版本以后,通过lua脚本解决争夺问题,实际上是redis利用其单线程的特性,用任务队列的方式解决多任务并发问题。
import (
 "context"
 "fmt"
 "GitHub.com/go-redis/redis/v8"
 "net"
 "time"
)

func useLua(userid, prodid string) bool {
 //编写脚本 - 检查数值,是否够用,够用再减,否则返回减掉后的结果
 var luaScript = redis.NewScript(`
  local userid=KEYS[1];
  local prodid=KEYS[2];
  local qTKEy="sk:"..prodid..":qt";
  local userKey="sk:"..prodid..":user";
  local userExists=redis.call("sismember",userKey,userid);
  if tonumber(userExists)==1 then
   return 2;
  end
  local num=redis.call("get",qtKey);
  if tonumber(num)<=0 then
   return 0;
  else
   redis.call("decr",qtKey);
   redis.call("SAdd",userKey,userid);
  end
  return 1;
 `)
 //执行脚本
 n, err := luaScript.Run(ctx, DB, []string{userid, prodid}).Result()
 if err != nil {
  return false
 }
 switch n {
 case int64(0):
  fmt.Println("抢购结束")
  return false
 case int64(1):
  fmt.Println(userid, ":抢购成功")
  return true
 case int64(2):
  fmt.Println(userid, ":已经抢购了")
  return false
 default:
  fmt.Println("发生未知错误!")
  return false
 }
 return true
}

func main() {
 // 并发的版本
 for i := 0; i < 20; i++ {
  go func() {
   uuid := GenerateUUID()
   prodid := "1023"
   time.Sleep(10 * time.Second)
   useLua(uuid, prodid)
  }()
 }
 time.Sleep(15 * time.Second)
}

到此这篇关于Go+Lua解决Redis秒杀中库存与超卖问题的文章就介绍到这了,更多相关Go Lua Redis秒杀中库存与超卖内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

您可能感兴趣的文档:

--结束END--

本文标题: Go+Lua解决Redis秒杀中库存与超卖问题

本文链接: https://www.lsjlt.com/news/197920.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Go+Lua解决Redis秒杀中库存与超卖问题
    目录0、简介1、简单版2、解决超卖3、解决库存问题Lua0、简介 Go语言连接go-Redis进行数据库的连接,如果你对这部分尚不了解,建议你先学习这部分知识。另外,本秒杀主要解决两个问题,第一个就是超卖问题,另一个就是...
    99+
    2023-03-01
    GoLuaRedis秒杀 GoLuaRedis库存与超卖
  • 怎么使用Go和Lua解决Redis秒杀中库存与超卖问题
    这篇文章主要介绍“怎么使用Go和Lua解决Redis秒杀中库存与超卖问题”,在日常操作中,相信很多人在怎么使用Go和Lua解决Redis秒杀中库存与超卖问题问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么使...
    99+
    2023-07-05
  • 解决redis秒杀超卖的问题
    小编给大家分享一下解决redis秒杀超卖的问题,希望大家阅读完这篇文章后大所收获,下面让我们一起去探讨吧!首先,生成库存的计数量 public function kucun() { ...
    99+
    2022-10-18
  • Redis中秒杀场景下超时与超卖问题如何解决
    这篇文章主要介绍“Redis中秒杀场景下超时与超卖问题如何解决”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Redis中秒杀场景下超时与超卖问题如何解决”文章能帮助大家解决问题。超时1.redis连...
    99+
    2023-06-30
  • Redis分布式锁解决秒杀超卖问题
    目录分布式锁应用场景单体锁的分类分布式锁核心逻辑分布式锁实现的问题——死锁和解决Redis解决删除别人锁的问题分布式锁应用场景 秒杀环境下:订单服务从库存中心拿到库存数,如果库存总数大于0,则进...
    99+
    2022-07-13
    Redis秒杀超卖 Redis分布式锁
  • Redis中秒杀场景下超时与超卖问题的解决方案
    目录超时1.redis连接超时原因2.解决方法超卖1.秒杀超卖现象2.解决方案(1)利用乐观锁淘汰用户,解决超卖问题(2)、使用reids的 watch + multi + setn...
    99+
    2022-11-13
  • 如何解决Redis高并发防止秒杀超卖的问题
    这篇文章主要介绍了如何解决Redis高并发防止秒杀超卖的问题,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。1:解决思路将活动写入 redis 中,通过 redis 自减指令扣...
    99+
    2023-06-25
  • Redis如何解决库存超卖问题
    这篇文章主要介绍“Redis如何解决库存超卖问题”,在日常操作中,相信很多人在Redis如何解决库存超卖问题问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Redis如何解决库存超卖问题”的疑惑有所帮助!接下来...
    99+
    2023-06-06
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作