欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Go+Lua解決Redis秒殺中庫(kù)存與超賣問(wèn)題

 更新時(shí)間:2023年03月01日 09:41:51   作者:ccgkk  
本文主要介紹了Go+Lua解決Redis秒殺中庫(kù)存與超賣問(wèn)題,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

0、簡(jiǎn)介

  • Go語(yǔ)言連接go-redis進(jìn)行數(shù)據(jù)庫(kù)的連接,如果你對(duì)這部分尚不了解,建議你先學(xué)習(xí)這部分知識(shí)。
  • 另外,本秒殺主要解決兩個(gè)問(wèn)題,第一個(gè)就是超賣問(wèn)題,另一個(gè)就是庫(kù)存問(wèn)題。
  • 沒(méi)有設(shè)計(jì)專門(mén)的頁(yè)面來(lái)模擬并發(fā),我們直接使用gorountine,在調(diào)用請(qǐng)求前停留10s。
  • 針對(duì)超賣問(wèn)題,引入go-redis的watch搭配事務(wù)處理即可【相當(dāng)于樂(lè)觀鎖】。

而針對(duì)庫(kù)存的問(wèn)題較為麻煩一點(diǎn),需要使用Lua編輯腳本,但是你無(wú)需在自己的機(jī)器上下載lua的編譯環(huán)境,go提供了其相關(guān)的支持。針對(duì)這一部分,不用慌張,其基本架構(gòu)如下:

1、簡(jiǎn)單版

面對(duì)并發(fā)的情況下會(huì)出現(xiàn)超賣的情況,redis數(shù)據(jù)庫(kù)中會(huì)出現(xiàn)負(fù)值的情況。即使你在操作之前進(jìn)行了數(shù)據(jù)的判斷。

func MsCode(uuid, prodid string) bool { ?
? ?// 1、對(duì)uuid和prodid進(jìn)行非空判斷 ?
? ?if uuid == "" || prodid == "" { ?
? ? ? return false ?
? ?} ?
??
? ?//2、獲取連接 ?
? ?rdb := DB ?
??
? ?//3、拼接key ?
? ?kcKey := "kc:" + prodid + ":qt" ?
? ?userKey := "sk:" + prodid + ":user" ?
??
? ?//4、獲取庫(kù)存 ?
? ?str, err := rdb.Get(ctx, kcKey).Result() ?
? ?if err != nil { ?
? ? ? fmt.Println(err) ?
? ? ? fmt.Println("秒殺還未開(kāi)始.......") ?
? ? ? return false ?
? ?} ?
??
? ?// 5、判斷用戶是否重復(fù)秒殺操作 ?
? ?flag, err := rdb.SIsMember(ctx, userKey, userKey).Result() ?
? ?if err != nil { ?
? ? ? fmt.Println(err) ?
? ?} ?
? ?if flag { ?
? ? ? fmt.Println("你已經(jīng)參加了秒殺,無(wú)法再次參加。。。。") ?
? ? ? return false ?
? ?} ?
??
? ?// 6、判斷商品數(shù)量,如果庫(kù)存數(shù)量小于1,秒殺結(jié)束 ?
? ?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("秒殺結(jié)束,請(qǐng)下次再來(lái)吧。。。。") ?
? ? ? return false ?
? ?} ?
??
? ?// 7、秒殺過(guò)程 ?
? ?// 7.1、庫(kù)存減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() {
?? ?// 并發(fā)的版本
?? ?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進(jìn)行監(jiān)視key,關(guān)鍵部分如下。但是這樣會(huì)造成一個(gè)問(wèn)題,就是搶購(gòu)不完,會(huì)有一些庫(kù)存,但是又有人沒(méi)有搶到。

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("搶購(gòu)結(jié)束了!請(qǐng)下次早點(diǎn)來(lái)。。。。")  
   }  
   _, 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、解決庫(kù)存問(wèn)題Lua

Lua操作redis能夠比較好的解決這個(gè)問(wèn)題。因?yàn)閞edis中使用watch是使用了悲觀鎖的形態(tài),而悲觀鎖會(huì)自然得造成庫(kù)存問(wèn)題,因此要使用樂(lè)觀鎖。而redis天然不支持樂(lè)觀鎖,基于此,需要時(shí)lua來(lái)編寫(xiě)相關(guān)腳本。其主要有以下優(yōu)勢(shì):

  • 將復(fù)雜的或者多步的redis操作,寫(xiě)為一個(gè)腳本,一次提交給redis執(zhí)行,減少反復(fù)連接redis的次數(shù)。提升性能。
  • luan腳本類似redis事務(wù),有一定的原子性,不會(huì)被其他命令插隊(duì),可以完成一些redis事務(wù)性的操作。
  • redis的lua腳本功能,只有在redis2.6以上的版本才可以使用。
  • 利用lua腳本淘汰用戶,解決超賣問(wèn)題。
  • redis2.6版本以后,通過(guò)lua腳本解決爭(zhēng)奪問(wèn)題,實(shí)際上是redis利用其單線程的特性,用任務(wù)隊(duì)列的方式解決多任務(wù)并發(fā)問(wèn)題。
import (
?? ?"context"
?? ?"fmt"
?? ?"github.com/go-redis/redis/v8"
?? ?"net"
?? ?"time"
)

func useLua(userid, prodid string) bool {
?? ?//編寫(xiě)腳本 - 檢查數(shù)值,是否夠用,夠用再減,否則返回減掉后的結(jié)果
?? ?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;
?? ?`)
?? ?//執(zhí)行腳本
?? ?n, err := luaScript.Run(ctx, DB, []string{userid, prodid}).Result()
?? ?if err != nil {
?? ??? ?return false
?? ?}
?? ?switch n {
?? ?case int64(0):
?? ??? ?fmt.Println("搶購(gòu)結(jié)束")
?? ??? ?return false
?? ?case int64(1):
?? ??? ?fmt.Println(userid, ":搶購(gòu)成功")
?? ??? ?return true
?? ?case int64(2):
?? ??? ?fmt.Println(userid, ":已經(jīng)搶購(gòu)了")
?? ??? ?return false
?? ?default:
?? ??? ?fmt.Println("發(fā)生未知錯(cuò)誤!")
?? ??? ?return false
?? ?}
?? ?return true
}

func main() {
?? ?// 并發(fā)的版本
?? ?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)
}

到此這篇關(guān)于Go+Lua解決Redis秒殺中庫(kù)存與超賣問(wèn)題的文章就介紹到這了,更多相關(guān)Go Lua Redis秒殺中庫(kù)存與超賣內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • golang?gorm學(xué)習(xí)之如何指定數(shù)據(jù)表

    golang?gorm學(xué)習(xí)之如何指定數(shù)據(jù)表

    在sql中首先要指定是從哪張表中查詢,所以這篇文章小編就來(lái)帶大家一起看一下gorm是如何根據(jù)model來(lái)自動(dòng)解析表名的,感興趣的小伙伴可以了解下
    2023-08-08
  • golang協(xié)程設(shè)計(jì)及調(diào)度原理

    golang協(xié)程設(shè)計(jì)及調(diào)度原理

    這篇文章主要介紹了golang協(xié)程設(shè)計(jì)及調(diào)度原理,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,感興趣的小伙伴可以參考一下
    2022-06-06
  • Go 語(yǔ)言中關(guān)于接口的三個(gè)

    Go 語(yǔ)言中關(guān)于接口的三個(gè)

    這篇文章主要介紹了Go 語(yǔ)言中關(guān)于接口的三個(gè)"潛規(guī)則",本文通過(guò)實(shí)例代碼相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-06-06
  • Golang實(shí)踐筆錄之讀取yaml配置文件

    Golang實(shí)踐筆錄之讀取yaml配置文件

    YAML是YAML?Ain't?a?Markup?Language的縮寫(xiě),YAML不是一種標(biāo)記語(yǔ)言,相比JSON格式的方便,這篇文章主要給大家介紹了關(guān)于Golang實(shí)踐筆錄之讀取yaml配置文件的相關(guān)資料,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2024-01-01
  • golang copy函數(shù)使用的坑

    golang copy函數(shù)使用的坑

    本文主要介紹了golang copy函數(shù)使用的坑,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-04-04
  • 深入理解Golang的反射reflect示例

    深入理解Golang的反射reflect示例

    本文主要介紹了Golang的反射reflect示例,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-01-01
  • 一文帶你了解Go語(yǔ)言中鎖的實(shí)現(xiàn)

    一文帶你了解Go語(yǔ)言中鎖的實(shí)現(xiàn)

    這篇文章主要帶大家一起學(xué)習(xí)一下go鎖和讀寫(xiě)鎖的總結(jié)文檔,?主要從"參考"部分的文章結(jié)合源碼學(xué)習(xí),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-03-03
  • Golang語(yǔ)言如何讀取http.Request中body的內(nèi)容

    Golang語(yǔ)言如何讀取http.Request中body的內(nèi)容

    這篇文章主要介紹了Golang語(yǔ)言如何讀取http.Request中body的內(nèi)容問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-03-03
  • 阿里云go開(kāi)發(fā)環(huán)境搭建過(guò)程

    阿里云go開(kāi)發(fā)環(huán)境搭建過(guò)程

    這篇文章主要介紹了阿里云go開(kāi)發(fā)環(huán)境搭建過(guò)程,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下
    2018-02-02
  • golang操作rocketmq的示例代碼

    golang操作rocketmq的示例代碼

    這篇文章主要介紹了golang操作rocketmq的示例代碼,代碼簡(jiǎn)單易懂,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-04-04

最新評(píng)論