婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Redis鎖完美解決高并發秒殺問題

Redis鎖完美解決高并發秒殺問題

熱門標簽:貴州電銷卡外呼系統 鄭州人工智能電銷機器人系統 十堰營銷電銷機器人哪家便宜 北京400電話辦理收費標準 超呼電話機器人 魔獸2青云地圖標注 日本中國地圖標注 宿遷便宜外呼系統平臺 山東外呼銷售系統招商

場景:一家網上商城做商品限量秒殺。

1 單機環境下的鎖

將商品的數量存到Redis中。每個用戶搶購前都需要到Redis中查詢商品數量(代替mysql數據庫。不考慮事務),如果商品數量大于0,則證明商品有庫存。然后我們在進行庫存扣減和接下來的操作。因為多線程并發問題,我們不得不在get()方法內部使用同步代碼塊。這樣可以保證查詢庫存和減庫存操作的原子性。

package springbootdemo.demo.controller;
/*
 * @auther 頂風少年
 * @mail dfsn19970313@foxmail.com
 * @date 2020-01-13 11:19
 * @notify
 * @version 1.0
 */

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class RedisLock  {

    @Autowired
    private RedisTemplateString, String> redisTemplate;

    @GetMapping(value = "buy")
    public String get() {
        synchronized (this) {
            String phone = redisTemplate.opsForValue().get("phone");
            Integer count = Integer.valueOf(phone);
            if (count > 0) {
                redisTemplate.opsForValue().set("phone", String.valueOf(count - 1));
                System.out.println("搶到了" + count + "號商品");
            }return "";
        }
    }
}

2 分布式情況下使用Redis鎖。

但是由于業務上升,并發數量變大。公司不得不將原有系統復制一份,放到新的服務器。然后使用nginx做負載均衡。為了模擬高并發環境這里使用了 Apache JMeter工具。

很明顯,現在的線程鎖不管用了。于是我們需要換一把鎖,這把鎖必須和兩套系統沒有任何的耦合度。

使用Redies的API如果key不存在,則設置一個key。這個key就是我們現在使用的一把鎖。每個線程到此處,先設置鎖,如果設置鎖失敗,則表明當前有線程獲取到了鎖,就返回。最后我們為了減庫存和其他業務拋出異常,而沒有釋放鎖。把釋放鎖的操作放到了finally代碼塊中。看起來是比較完美了。

package springbootdemo.demo.controller;
/*
 * @auther 頂風少年
 * @mail dfsn19970313@foxmail.com
 * @date 2020-01-13 11:19
 * @notify
 * @version 1.0
 */

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class RedisLock {

    @Autowired
    private RedisTemplateString, String> redisTemplate;

    @GetMapping(value = "buy")
    public String get() {
        Boolean phoneLock = redisTemplate.opsForValue().setIfAbsent("phoneLock", "");
        if (!phoneLock) {
            return "";
        }
        try{
            String phone = redisTemplate.opsForValue().get("phone");
            Integer count = Integer.valueOf(phone);
            if (count > 0) {
                redisTemplate.opsForValue().set("phone", String.valueOf(count - 1));
                System.out.println("搶到了" + count + "號商品");
            }
        }finally {
            redisTemplate.delete("phoneLock");
        }
        return "";
    }
}

3 一臺服務宕機,導致無法釋放鎖

如果try中拋出了異常,進入finally,這把鎖依然會釋放,不會影響其他線程獲取鎖,那么如果在finally也拋出了異常,或者在finally中服務直接關閉了,那其他的服務再也獲取不到鎖。最終導致商品賣不出去。

package springbootdemo.demo.controller;
/*
 * @auther 頂風少年
 * @mail dfsn19970313@foxmail.com
 * @date 2020-01-13 11:19
 * @notify
 * @version 1.0
 */

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class RedisLock {

    @Autowired
    private RedisTemplateString, String> redisTemplate;

    @GetMapping(value = "buy")
    public String get() {
        int i = 0;
        Boolean phoneLock = redisTemplate.opsForValue().setIfAbsent("phoneLock", "");
        if (!phoneLock) {
            return "";
        }
        try {
            String phone = redisTemplate.opsForValue().get("phone");
            Integer count = Integer.valueOf(phone);
            if (count > 0) {
                i = count;
                redisTemplate.opsForValue().set("phone", String.valueOf(count - 1));
                System.out.println("搶到了" + count + "號商品");
            }
        } finally {
            if (i == 20) {
                System.exit(0);
            }
            redisTemplate.delete("phoneLock");
        }
        return "";
    }
}

4 給每一把鎖加上過期時間

問題就出現在如果出現意外,這把鎖無法釋放。這里我們在引入Redis的API,對key進行過期時間的設置。這樣如果拿到鎖的線程,在任何情況下沒有來得及釋放鎖,當Redis的key時間到,也會自動釋放鎖。但是這樣還是存在問題

如果在key過期后,鎖釋放了,但是當前線程沒有執行完畢。那么其他線程就會拿到鎖,繼續搶購商品,而這個較慢的線程則會在執行完畢后,釋放別人的鎖。導致鎖失效!

package springbootdemo.demo.controller;
/*
 * @auther 頂風少年
 * @mail dfsn19970313@foxmail.com
 * @date 2020-01-13 11:19
 * @notify
 * @version 1.0
 */

import javafx.concurrent.Task;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.Timer;
import java.util.TimerTask;
import java.util.concurrent.TimeUnit;

@RestController
public class RedisLock {

    @Autowired
    private RedisTemplateString, String> redisTemplate;

    @GetMapping(value = "buy")
    public String get() {
        Boolean phoneLock = redisTemplate.opsForValue().setIfAbsent("phoneLock", "", 3, TimeUnit.SECONDS);
        if (!phoneLock) {
            return "";
        }
        try {
            String phone = redisTemplate.opsForValue().get("phone");
            Integer count = Integer.valueOf(phone);
            if (count > 0) {
                try {
                    Thread.sleep(99999999999L);
                } catch (Exception e) {

                }
                redisTemplate.opsForValue().set("phone", String.valueOf(count - 1));
                System.out.println("搶到了" + count + "號商品");
            }
        } finally {
          
            redisTemplate.delete("phoneLock");
        }
        return "";
    }
}

5延長鎖的過期時間,解決鎖失效

問題的出現就是,當一條線程的key已經過期,但是這個線程的任務確確實實沒有執行完畢,這個交易沒有結束。但是鎖沒了?,F在我們必須對鎖的時間進行延長。在判斷商品有庫存時,第一時間創建一個線程不停的給key續命,

防止key過期。然后在交易結束后,停止定時器,釋放鎖。

package springbootdemo.demo.controller;
/*
 * @auther 頂風少年
 * @mail dfsn19970313@foxmail.com
 * @date 2020-01-13 11:19
 * @notify
 * @version 1.0
 */

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.Timer;
import java.util.TimerTask;
import java.util.concurrent.TimeUnit;

@RestController
public class RedisLock {

    @Autowired
    private RedisTemplateString, String> redisTemplate;

    @GetMapping(value = "buy")
    public String get() {
        Boolean phoneLock = redisTemplate.opsForValue().setIfAbsent("phoneLock", "", 3, TimeUnit.SECONDS);
        if (!phoneLock) {
            return "";
        }
        Timer timer = null;
        try {
            String phone = redisTemplate.opsForValue().get("phone");
            Integer count = Integer.valueOf(phone);
            if (count > 0) {
                timer = new Timer();
                timer.schedule(new TimerTask() {
                    @Override
                    public void run() {
                        redisTemplate.opsForValue().set("phoneLock", "", 3, TimeUnit.SECONDS);
                    }
                }, 0, 1);

                redisTemplate.opsForValue().set("phone", String.valueOf(count - 1));
                System.out.println("搶到了" + count + "號商品");
            }
        } finally {
            if (timer != null) {
                timer.cancel();
            }
            redisTemplate.delete("phoneLock");
        }
        return "";
    }
}

6 使用Redisson簡化代碼

在步驟5我們的代碼已經很完善了,不會出現高并發問題。但是代碼確過于冗余,我們為了使用Redis鎖,我們需要設置一個定長的key,然后當購買完成后,將key刪除。但為了防止key提前過期,我們不得不新增一個線程執行定時任務。下面我們可以使用Redissson框架簡化代碼。getLock()方法代替了Redis的setIfAbsent(),lock()設置過期時間。最終我們在交易結束后釋放鎖。延長鎖的操作則有Redisson框架替我們完成,它會使用輪詢去查看key是否過期,

在交易沒有完成時,自動重設Redis的key過期時間

package springbootdemo.demo.controller;
/*
 * @auther 頂風少年
 * @mail dfsn19970313@foxmail.com
 * @date 2020-01-13 11:19
 * @notify
 * @version 1.0
 */

import org.redisson.Redisson;
import org.redisson.api.RLock;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.Timer;
import java.util.TimerTask;
import java.util.concurrent.TimeUnit;

@RestController
public class RedissonLock {

    @Autowired
    private RedisTemplateString, String> redisTemplate;

    @Autowired
    private Redisson redisson;

    @GetMapping(value = "buy2")
    public String get() {
        RLock phoneLock = redisson.getLock("phoneLock");
        phoneLock.lock(3, TimeUnit.SECONDS);
        try {
            String phone = redisTemplate.opsForValue().get("phone");
            Integer count = Integer.valueOf(phone);
            if (count > 0) {
                redisTemplate.opsForValue().set("phone", String.valueOf(count - 1));
                System.out.println("搶到了" + count + "號商品");
            }
        } finally {
            phoneLock.unlock();
        }
        return "";
    }
}

到此這篇關于Redis鎖完美解決高并發秒殺問題的文章就介紹到這了,更多相關Redis鎖高并發秒殺內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • Redis瞬時高并發秒殺方案總結
  • php結合redis實現高并發下的搶購、秒殺功能的實例

標簽:北京 臺州 果洛 吉安 朝陽 大慶 楊凌 江蘇

巨人網絡通訊聲明:本文標題《Redis鎖完美解決高并發秒殺問題》,本文關鍵詞  Redis,鎖,完美,解決,高并發,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Redis鎖完美解決高并發秒殺問題》相關的同類信息!
  • 本頁收集關于Redis鎖完美解決高并發秒殺問題的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    欧美主播一区二区三区| 日韩精品电影一区亚洲| 亚洲男帅同性gay1069| 日韩在线卡一卡二| 在线精品视频免费播放| 久久久av毛片精品| 国产一区二区在线看| 欧美一区三区二区| 午夜免费久久看| 欧美日韩国产一级| 天天av天天翘天天综合网色鬼国产 | 国产.精品.日韩.另类.中文.在线.播放| 欧美一级二级在线观看| 激情久久久久久久久久久久久久久久| 日韩一区二区中文字幕| 久久99国产精品尤物| 日韩欧美在线综合网| 国产美女av一区二区三区| 亚洲国产精品99久久久久久久久 | 理论电影国产精品| 久久综合九色综合欧美就去吻| 国内外成人在线视频| 久久久久国产精品厨房| 91免费国产视频网站| 日韩电影在线看| 国产女同互慰高潮91漫画| 91丨九色丨国产丨porny| 亚洲最大的成人av| 在线综合视频播放| 福利一区在线观看| 亚洲午夜久久久久久久久电影网| 欧美日韩国产大片| 国产剧情一区二区三区| 亚洲视频免费在线观看| 91精品婷婷国产综合久久性色| 国产精一区二区三区| 18成人在线视频| 欧美美女激情18p| 韩国一区二区视频| 亚洲小说欧美激情另类| 久久久久亚洲蜜桃| 欧美日韩国产一区二区三区地区| 久久精品二区亚洲w码| 国产精品久久久一区麻豆最新章节| 911国产精品| 不卡大黄网站免费看| 日本欧美大码aⅴ在线播放| 樱桃国产成人精品视频| 久久尤物电影视频在线观看| 欧美性猛片xxxx免费看久爱| 国产精品1区2区| 日韩成人精品在线观看| 中文字幕欧美一| 久久久久国产精品麻豆ai换脸| 欧美日韩免费一区二区三区视频| 丁香激情综合五月| 极品美女销魂一区二区三区免费 | 午夜av一区二区| 中文字幕一区二区三区精华液| 久久久精品2019中文字幕之3| 51久久夜色精品国产麻豆| 色偷偷久久一区二区三区| 国产福利91精品| 狠狠色综合色综合网络| 青青草伊人久久| 亚洲成人免费影院| 亚洲电影中文字幕在线观看| 亚洲精品亚洲人成人网| 亚洲欧美综合在线精品| 国产精品美女久久久久久久网站| 日本一区二区三级电影在线观看| 久久品道一品道久久精品| 日韩精品最新网址| 精品国产一区二区三区不卡 | 中文在线资源观看网站视频免费不卡 | 欧美日韩不卡一区二区| 在线精品视频一区二区| 欧美日韩高清不卡| 欧美一级艳片视频免费观看| 欧美不卡视频一区| 国产欧美视频在线观看| 国产喂奶挤奶一区二区三区| 国产欧美一区二区三区沐欲| 国产精品不卡在线| 尤物在线观看一区| 久久精品av麻豆的观看方式| 国产高清一区日本| 色中色一区二区| 欧美精品乱码久久久久久| 日韩一级成人av| 国产欧美久久久精品影院| 亚洲少妇最新在线视频| 亚洲超碰精品一区二区| 久久av老司机精品网站导航| 成人高清免费观看| 欧美私人免费视频| 欧美xxxxxxxxx| 亚洲欧美自拍偷拍| 日韩中文字幕区一区有砖一区| 美腿丝袜亚洲色图| 国产精品一区二区久久不卡| 在线观看不卡一区| 精品成人一区二区三区四区| 一区二区三区在线观看视频| 久久爱另类一区二区小说| 91在线免费看| 91精品国产免费| 亚洲欧洲在线观看av| 日本免费新一区视频| 91日韩在线专区| 日韩一区二区三区电影| 亚洲女人的天堂| 极品少妇xxxx精品少妇偷拍 | 制服丝袜av成人在线看| 18成人在线观看| 久久国产精品第一页| 91麻豆精品在线观看| 欧美mv日韩mv| 亚洲午夜久久久久久久久久久| 成人性生交大合| 日韩一卡二卡三卡四卡| 亚洲免费高清视频在线| 国产成人夜色高潮福利影视| 337p亚洲精品色噜噜狠狠| 亚洲影视资源网| 91蜜桃婷婷狠狠久久综合9色| 26uuu亚洲综合色| 天堂影院一区二区| 欧美写真视频网站| 亚洲欧美电影院| av福利精品导航| 国产亚洲一区二区在线观看| 蜜桃av噜噜一区| 91.com在线观看| 一区二区三区免费网站| 波多野结衣在线aⅴ中文字幕不卡| 欧美大黄免费观看| 麻豆成人在线观看| 日韩一区二区三区在线| 亚洲一区二区在线免费观看视频| 99久久99久久综合| 亚洲天堂中文字幕| 91论坛在线播放| 亚洲女人****多毛耸耸8| 色婷婷综合五月| 一级特黄大欧美久久久| 欧美在线制服丝袜| 香蕉成人伊视频在线观看| 欧美男女性生活在线直播观看| 一区二区三区四区在线播放| 欧美亚洲禁片免费| 丝袜亚洲另类欧美| 欧美精品黑人性xxxx| 蜜臀av性久久久久av蜜臀妖精 | 精品久久久久久久久久久久久久久| 日韩国产成人精品| 日韩一级免费观看| 久久99精品久久只有精品| 2021久久国产精品不只是精品| 精品一区二区三区在线播放视频| 日韩欧美国产午夜精品| 国产精品亚洲专一区二区三区| 中文幕一区二区三区久久蜜桃| 成人av在线资源网| 亚洲午夜视频在线观看| 91精品国产乱| 国产精品夜夜爽| 亚洲综合激情另类小说区| 在线不卡一区二区| 国产成人啪午夜精品网站男同| 国产精品区一区二区三| 欧美日韩一区二区三区在线看| 美国三级日本三级久久99| 欧美激情一区二区在线| 欧美精品在线观看播放| 麻豆成人av在线| 国产精品免费视频观看| 69av一区二区三区| 国产成人av一区二区三区在线| 中文字幕免费不卡在线| 欧美日韩电影在线播放| 国产成人免费视频网站高清观看视频 | 亚洲美女屁股眼交| 日韩一卡二卡三卡四卡| 91蜜桃在线观看| 国产在线精品免费| 亚洲一区二区综合| 国产欧美日韩精品一区| 欧美日韩亚洲综合在线| 成人性视频免费网站| 老司机一区二区| 亚洲欧美另类综合偷拍| 久久免费视频色| 欧美疯狂性受xxxxx喷水图片| 国产麻豆精品在线| 蜜臀av一区二区在线免费观看| 中文字幕欧美区| 欧美成人乱码一区二区三区| 色婷婷av久久久久久久| 国产一区不卡精品|