亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Redis實(shí)戰(zhàn)之Redis實(shí)現(xiàn)異步秒殺優(yōu)化詳解

 更新時(shí)間:2023年09月14日 09:08:31   作者:明礬java  
這篇文章主要給大家介紹了Redis實(shí)戰(zhàn)之Redis實(shí)現(xiàn)異步秒殺優(yōu)化方法,文章通過圖片和代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,感興趣的同學(xué)可以自己動(dòng)手試一下

秒殺優(yōu)化-異步秒殺思路

未優(yōu)化的思路

當(dāng)用戶發(fā)起請(qǐng)求,此時(shí)會(huì)請(qǐng)求nginx,nginx會(huì)訪問到tomcat,而tomcat中的程序,會(huì)進(jìn)行串行操作,分成如下幾個(gè)步驟

1、查詢優(yōu)惠卷

2、判斷秒殺庫存是否足夠

3、查詢訂單

4、校驗(yàn)是否是一人一單

5、扣減庫存

6、創(chuàng)建訂單

在這六步操作中,又有很多操作是要去操作數(shù)據(jù)庫的,而且還是一個(gè)線程串行執(zhí)行, 這樣就會(huì)導(dǎo)致我們的程序執(zhí)行的很慢

優(yōu)化方案

我們將耗時(shí)比較短的邏輯判斷放入到redis中,比如是否庫存足夠,比如是否一人一單,這樣的操作,只要這種邏輯可以完成,就意味著我們是一定可以下單完成的,我們只需要進(jìn)行快速的邏輯判斷,根本就不用等下單邏輯走完,我們直接給用戶返回成功, 再在后臺(tái)開一個(gè)線程,后臺(tái)線程慢慢的去執(zhí)行queue里邊的消息,即不追求時(shí)效性,讓用戶先成功下單,后續(xù)再完善數(shù)據(jù)庫數(shù)據(jù)

整體思路

用戶下單之后,判斷庫存是否充足只需要到redis中去根據(jù)key找對(duì)應(yīng)的value是否大于0即可,如果不充足,則直接結(jié)束,如果充足,繼續(xù)在redis中判斷用戶是否可以下單,如果set集合中沒有這條數(shù)據(jù),說明他可以下單,如果set集合中沒有這條記錄,則將userId和優(yōu)惠卷存入到redis中,并且返回0,整個(gè)過程需要保證是原子性的,我們可以使用lua來操作

當(dāng)以上判斷邏輯走完之后,我們可以判斷當(dāng)前redis中返回的結(jié)果是否是0 ,如果是0,則表示可以下單,則將之前說的信息存入到到queue中去,然后返回,然后再來個(gè)線程異步的下單,前端可以通過返回的訂單id來判斷是否下單成功。

難點(diǎn)

  • 怎么在redis中去快速校驗(yàn)一人一單,還有庫存判斷
  • 由于我們校驗(yàn)和tomct下單是兩個(gè)線程,那么我們?nèi)绾沃赖降啄膫€(gè)單他最后是否成功,或者是下單完成,為了完成這件事我們?cè)趓edis操作完之后,我們會(huì)將一些信息返回給前端,同時(shí)也會(huì)把這些信息丟到異步queue中去,后續(xù)操作中,可以通過這個(gè)id來查詢我們tomcat中的下單邏輯是否完成了。

代碼實(shí)現(xiàn)

需求:

  • 新增秒殺優(yōu)惠券的同時(shí),將優(yōu)惠券信息,優(yōu)惠券id和庫存信息保存到Redis中
  • 基于Lua腳本,判斷秒殺庫存、一人一單,決定用戶是否搶購(gòu)成功
  • 如果搶購(gòu)成功,將優(yōu)惠券id和用戶id封裝后存入阻塞隊(duì)列
  • 開啟線程任務(wù),不斷從阻塞隊(duì)列中獲取信息,實(shí)現(xiàn)異步下單功能

新增優(yōu)惠券,將優(yōu)惠券信息入庫并寫入redis

@Override
    @Transactional
    public void addSeckillVoucher(Voucher voucher) {
        // 保存優(yōu)惠券
        save(voucher);
        // 保存秒殺信息
        SeckillVoucher seckillVoucher = new SeckillVoucher();
        seckillVoucher.setVoucherId(voucher.getId());
        seckillVoucher.setStock(voucher.getStock());
        seckillVoucher.setBeginTime(voucher.getBeginTime());
        seckillVoucher.setEndTime(voucher.getEndTime());
        seckillVoucherService.save(seckillVoucher);
//存入redis
        stringRedisTemplate.opsForValue().setIfAbsent(SECKILL_STOCK_KEY + voucher.getId(), voucher.getStock().toString());
    }

判斷秒殺庫存、一人一單,決定用戶是否搶購(gòu)成功,考慮到操作的原子性,采用lua腳本完成這一連串的操作

---
--- Generated by EmmyLua(https://github.com/EmmyLua)
--- Created by Lenovo.
--- DateTime: 2023/9/5 20:57
---
-- 1.參數(shù)列表
-- 1.1.優(yōu)惠券id
local voucherId = ARGV[1]
-- 1.2.用戶id
local userId = ARGV[2]
---- 1.3.訂單id
local orderId = ARGV[3]
-- 2.數(shù)據(jù)key
-- 2.1.庫存key
local stockKey = 'seckill:stock:' .. voucherId
---- 2.2.訂單key
local orderKey = 'seckill:order:' .. voucherId
-- 3.腳本業(yè)務(wù)
-- 3.1.判斷庫存是否充足 get stockKey
if(tonumber(redis.call('get', stockKey)) <= 0) then
    -- 3.2.庫存不足,返回1
    return 1
end
-- 3.2.判斷用戶是否下單 SISMEMBER orderKey userId
if(redis.call('sismember', orderKey, userId) == 1) then
    -- 3.3.存在,說明是重復(fù)下單,返回2
    return 2
end
-- 3.4.扣庫存 incrby stockKey -1
redis.call('incrby', stockKey, -1)
-- 3.5.下單(保存用戶)sadd orderKey userId
redis.call('sadd', orderKey, userId)
---- 3.6.發(fā)送消息到隊(duì)列中, XADD stream.orders * k1 v1 k2 v2 ...
redis.call('xadd', 'stream.orders', '*', 'userId', userId, 'voucherId', voucherId, 'id', orderId)
return 0

執(zhí)行l(wèi)ua腳本,判斷是否搶購(gòu)成功,如果搶購(gòu)成功,要放入堵塞隊(duì)列中

@Override
    public Result seckillVoucher(Long voucherId) {
        SeckillVoucher seckillVoucher = seckillVoucherService.getById(voucherId);
        //判斷是否開始,開始時(shí)間如果在當(dāng)前時(shí)間之后就是尚未開始
        if (seckillVoucher.getBeginTime().isAfter(LocalDateTime.now())) {
            return Result.fail("秒殺尚未開始");
        }
        //判斷是否結(jié)束,結(jié)束時(shí)間如果在當(dāng)前時(shí)間之前就是已經(jīng)結(jié)束
        if (seckillVoucher.getEndTime().isBefore(LocalDateTime.now())) {
            return Result.fail("秒殺已經(jīng)結(jié)束");
        }
        Long userId = UserHolder.getUser().getId();
        long orderId = new RedisIdWorker(stringRedisTemplate).nextId("order");
        Long execute = stringRedisTemplate.execute(SILLL_SCRIPT,
                Collections.emptyList(),
                voucherId.toString(), userId.toString(), String.valueOf(orderId)
        );
        int r = execute.intValue();
        if (r != 0) {
            return Result.fail(r == 1 ? "庫存不足" : "不能重復(fù)下單");
        }
        VoucherOrder voucherOrder = new VoucherOrder();
        //訂單id
        voucherOrder.setUserId(userId);
        voucherOrder.setVoucherId(voucherId);
        voucherOrder.setId(orderId);
        //將訂單信息放入阻塞隊(duì)列
        orderTakes.add(voucherOrder);
        return Result.ok(orderId);
    }

定義線程內(nèi)部類,不斷從堵塞隊(duì)列中讀取訂單

//從阻塞隊(duì)列里面取訂單信息
    private class voucherOrderHander implements Runnable {
        @Override
        public void run() {
            while (true) {
                try {
                    VoucherOrder take = orderTakes.take();
                    handleVoucherOrder(take);
                } catch (Exception e) {
                    log.error("異常信息如下", e);
                }
            }
        }

獲取訂單信息的具體方法,這里依然加了分布式鎖,是為了保險(xiǎn)起見

 private void handleVoucherOrder(VoucherOrder take) {
            Long userId = take.getId();
            //創(chuàng)建鎖對(duì)象
            RLock lock = redissonClient.getLock("lock:order:" + userId);
            //嘗試獲取鎖
            boolean isLock = lock.tryLock();
            //獲取鎖失敗
            if (!isLock) {
                log.error("不允許重復(fù)下單");
                return;
            }
            try {
                voucherOrderService.createVoucherOrder(take);
            } finally {
                //釋放鎖
                lock.unlock();
            }
        }
    }

這里又有一個(gè)問題,就是我們訂單信息入庫應(yīng)該是在該類對(duì)象被創(chuàng)建的時(shí)候就要開啟線程在堵塞隊(duì)列等待讀取是否有訂單信息,然后順利入庫,所以我們用了aop的@PostConstruct,保證該對(duì)象被創(chuàng)建時(shí),線程也能順利創(chuàng)建,這里用了線程池來提交線程任務(wù)

@PostConstruct
    public void init() {
        SECKILL_ORDER_EXECUTOR.execute(new voucherOrderHander());
    }

完整代碼實(shí)現(xiàn)

@Service
public class VoucherOrderServiceImpl extends ServiceImpl<VoucherOrderMapper, VoucherOrder> implements IVoucherOrderService {
    @Autowired
    private ISeckillVoucherService seckillVoucherService;
    @Autowired
    private RedisIdWorker redisIdWorker;
    @Autowired
    private IVoucherOrderService voucherOrderService;
    @Autowired
    private StringRedisTemplate stringRedisTemplate;
    @Autowired
    private RedissonClient redissonClient;
    private static final DefaultRedisScript<Long> SILLL_SCRIPT;
    BlockingQueue<VoucherOrder> orderTakes = new ArrayBlockingQueue<>(1024 * 1024);
    //異步處理線程池
    private static final ExecutorService SECKILL_ORDER_EXECUTOR = Executors.newSingleThreadExecutor();
    static {
        SILLL_SCRIPT = new DefaultRedisScript<>();
        SILLL_SCRIPT.setLocation(new ClassPathResource("skill.lua"));
        SILLL_SCRIPT.setResultType(Long.class);
    }
    @PostConstruct
    public void init() {
        SECKILL_ORDER_EXECUTOR.execute(new voucherOrderHander());
    }
    //從阻塞隊(duì)列里面取用戶信息
    private class voucherOrderHander implements Runnable {
        @Override
        public void run() {
            while (true) {
                try {
                    VoucherOrder take = orderTakes.take();
                    handleVoucherOrder(take);
                } catch (Exception e) {
                    log.error("異常信息如下", e);
                }
            }
        }
        private void handleVoucherOrder(VoucherOrder take) {
            Long userId = take.getId();
            //創(chuàng)建鎖對(duì)象
            RLock lock = redissonClient.getLock("lock:order:" + userId);
            //嘗試獲取鎖
            boolean isLock = lock.tryLock();
            //獲取鎖失敗
            if (!isLock) {
                log.error("不允許重復(fù)下單");
                return;
            }
            try {
                voucherOrderService.createVoucherOrder(take);
            } finally {
                //釋放鎖
                lock.unlock();
            }
        }
    }
    @Override
    public Result seckillVoucher(Long voucherId) {
        SeckillVoucher seckillVoucher = seckillVoucherService.getById(voucherId);
        //判斷是否開始,開始時(shí)間如果在當(dāng)前時(shí)間之后就是尚未開始
        if (seckillVoucher.getBeginTime().isAfter(LocalDateTime.now())) {
            return Result.fail("秒殺尚未開始");
        }
        //判斷是否結(jié)束,結(jié)束時(shí)間如果在當(dāng)前時(shí)間之前就是已經(jīng)結(jié)束
        if (seckillVoucher.getEndTime().isBefore(LocalDateTime.now())) {
            return Result.fail("秒殺已經(jīng)結(jié)束");
        }
        Long userId = UserHolder.getUser().getId();
        long orderId = new RedisIdWorker(stringRedisTemplate).nextId("order");
        Long execute = stringRedisTemplate.execute(SILLL_SCRIPT,
                Collections.emptyList(),
                voucherId.toString(), userId.toString(), String.valueOf(orderId)
        );
        int r = execute.intValue();
        if (r != 0) {
            return Result.fail(r == 1 ? "庫存不足" : "不能重復(fù)下單");
        }
        VoucherOrder voucherOrder = new VoucherOrder();
        //訂單id
        voucherOrder.setUserId(userId);
        voucherOrder.setVoucherId(voucherId);
        voucherOrder.setId(orderId);
        //將訂單信息放入阻塞隊(duì)列
        orderTakes.add(voucherOrder);
        return Result.ok(orderId);
    }
@Transactional
    public void createVoucherOrder(VoucherOrder voucherOrder) {
        Long userId = voucherOrder.getUserId();
        // 5.1.查詢訂單
        int count = query().eq("user_id", userId).eq("voucher_id", voucherOrder.getVoucherId()).count();
        // 5.2.判斷是否存在
        if (count > 0) {
            // 用戶已經(jīng)購(gòu)買過了
            log.error("用戶已經(jīng)購(gòu)買過了");
            return;
        }
        // 6.扣減庫存
        boolean success = seckillVoucherService.update()
                .setSql("stock = stock - 1") // set stock = stock - 1
                .eq("voucher_id", voucherOrder.getVoucherId()).gt("stock", 0) // where id = ? and stock > 0
                .update();
        if (!success) {
            // 扣減失敗
            log.error("庫存不足");
            return;
        }
        save(voucherOrder);
    }

以上就是Redis實(shí)戰(zhàn)之Redis實(shí)現(xiàn)異步秒殺優(yōu)化詳解的詳細(xì)內(nèi)容,更多關(guān)于Redis實(shí)現(xiàn)異步秒殺優(yōu)化的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Redis實(shí)現(xiàn)登錄注冊(cè)的示例代碼

    Redis實(shí)現(xiàn)登錄注冊(cè)的示例代碼

    本文主要介紹了Redis實(shí)現(xiàn)登錄注冊(cè)的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-06-06
  • redis5.0以上基于密碼認(rèn)證的集群cluster方式

    redis5.0以上基于密碼認(rèn)證的集群cluster方式

    這篇文章主要介紹了redis5.0以上基于密碼認(rèn)證的集群cluster方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-11-11
  • 詳解如何利用Redis實(shí)現(xiàn)生成唯一ID

    詳解如何利用Redis實(shí)現(xiàn)生成唯一ID

    隨著下單流量逐漸上升,為了降低數(shù)據(jù)庫的訪問壓力,需要通過請(qǐng)求唯一ID+redis分布式鎖來防止接口重復(fù)提交。今天我們就一起來看探討一下,如何通過服務(wù)端來完成請(qǐng)求唯一?ID?的生成
    2022-11-11
  • Redis數(shù)據(jù)結(jié)構(gòu)之鏈表詳解

    Redis數(shù)據(jù)結(jié)構(gòu)之鏈表詳解

    大家好,本篇文章主要講的是Redis數(shù)據(jù)結(jié)構(gòu)之鏈表詳解,感興趣的同學(xué)趕快來看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽
    2021-12-12
  • Redis Cluster原理及配置詳解

    Redis Cluster原理及配置詳解

    這篇文章主要為大家介紹了Redis Cluster原理及配置詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-11-11
  • redis?for?windows?6.2.6安裝包最新步驟詳解

    redis?for?windows?6.2.6安裝包最新步驟詳解

    這篇文章主要介紹了redis?for?windows?6.2.6安裝包全網(wǎng)首發(fā),使用Windows計(jì)劃任務(wù)自動(dòng)運(yùn)行redis服務(wù),文章給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-04-04
  • Redis RDB技術(shù)底層原理詳解

    Redis RDB技術(shù)底層原理詳解

    為了使Redis在重啟之后仍能保證數(shù)據(jù)不丟失,需要將數(shù)據(jù)從內(nèi)存中以某種形式同步到硬盤中,這一過程就是持久化,本文重點(diǎn)給大家介紹Redis RDB技術(shù)底層原理實(shí)現(xiàn)方法,一起看看吧
    2021-09-09
  • Redis唯一ID生成器的實(shí)現(xiàn)

    Redis唯一ID生成器的實(shí)現(xiàn)

    本文主要介紹了Redis唯一ID生成器的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07
  • redis啟動(dòng)redis-server.exe閃退問題解決

    redis啟動(dòng)redis-server.exe閃退問題解決

    本文主要介紹了redis啟動(dòng)redis-server.exe閃退問題解決,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2025-02-02
  • Redis配置外網(wǎng)可訪問(redis遠(yuǎn)程連接不上)的方法

    Redis配置外網(wǎng)可訪問(redis遠(yuǎn)程連接不上)的方法

    默認(rèn)情況下,當(dāng)我們?cè)诓渴鹆藃edis服務(wù)之后,redis本身默認(rèn)只允許本地訪問。Redis服務(wù)端只允許它所在服務(wù)器上的客戶端訪問,如果Redis服務(wù)端和Redis客戶端不在同一個(gè)機(jī)器上,就要進(jìn)行配置。
    2022-12-12

最新評(píng)論