91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

怎么使用Go和Lua解決Redis秒殺中庫存與超賣問題

發布時間:2023-03-01 11:05:06 來源:億速云 閱讀:117 作者:iii 欄目:開發技術

這篇文章主要介紹“怎么使用Go和Lua解決Redis秒殺中庫存與超賣問題”,在日常操作中,相信很多人在怎么使用Go和Lua解決Redis秒殺中庫存與超賣問題問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”怎么使用Go和Lua解決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秒殺中庫存與超賣問題”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注億速云網站,小編會繼續努力為大家帶來更多實用的文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

昌吉市| 江油市| 东阿县| 马龙县| 武山县| 巴塘县| 岳阳县| 土默特左旗| 石城县| 疏勒县| 万年县| 长乐市| 托里县| 尤溪县| 桑日县| 山西省| 黔江区| 宜州市| 汶川县| 钦州市| 安宁市| 平果县| 济源市| 佛山市| 永靖县| 自贡市| 胶州市| 凤城市| 天柱县| 临泉县| 武清区| 伊通| 中山市| 抚宁县| 宁南县| 鹤岗市| 琼结县| 通州市| 常宁市| 临夏市| 涞源县|