主頁 > 後端開發 > SpringCloud(十一)- 秒殺 搶購

SpringCloud(十一)- 秒殺 搶購

2022-11-26 07:13:18 後端開發

1、流程圖

1.1 資料預熱

1.2 搶購

1.3 生成訂單 (發送訂單訊息)

1.4 訂單入庫 (監聽 消費訂單訊息)

1.5 查看訂單狀態

1.6 支付 (獲取支付鏈接 )

1.7 支付成功 微信回呼 (發送 支付成功訊息)

1.8 支付成功 回傳給前端成功 (監聽 支付成功訊息)

2、incr 和 setnx

2.1 incr

Redis Incr 命令將 key 中儲存的數字值增一, 如果 key 不存在,那么 key 的值會先被初始化為 0 ,然后再執行 INCR 操作,且將key的有效時間設定為長期有效 ,

2.1.1 常見使用場景

2.1.1.1 計數

我們可能常會統計網站頁面每天訪問量,通過incr命令在redis中設定key,每次增加1,設定24小時過期,

2.1.1.2 限流

日常的開放平臺API一般常有限流,利用redis的incr命令可以實作一般的限流操作,如限制某介面每分鐘請求次數上限1000次

2.1.1.3 冪等

MQ防止重復消費也可以利用INCR命令實作,如訂單防重,訂單5分鐘之內只能被消費一次,訂單號作為redis的key

2.2 sexnx

Redis使用setnx命令實作分布式鎖;

2.1.1 加鎖

版本一#但是這種方式的缺點是容易產生死鎖,因為客戶端有可能忘記解鎖,或者解鎖失敗,
setnx key value

版本二#給鎖增加了過期時間,避免出現死鎖,但這兩個命令不是原子的,第二步可能會失敗,依然無法避免死鎖問題,
setnx key value 
expire key seconds

版本三(推薦)#通過“set...nx...”命令,將加鎖、過期命令編排到一起,它們是原子操作了,可以避免死鎖,
set key value nx ex seconds

2.1.2 解鎖

解鎖就是洗掉代表鎖的那份資料,

del key

參考博客:https://blog.csdn.net/weixin_45525272/article/details/126562119


3、實作代碼

主要是搶購的業務;

3.1模塊分析

3.2 web模塊

3.2.1 application.yml

點擊查看代碼
# 埠
server:
  port: 8106

# 服務名
spring:
  application:
    name: edocmall-seckill-web

# redis 配置
  redis:
    host: 127.0.0.1
    port: 6379

# eureka 注冊中心的配置
eureka:
  client:
    service-url:
      defaultZone: http://127.0.0.1:8096/eureka  # 注冊中心的地址

# 秒殺搶購自定義配置
kh96:
  seckill:
    buy-user-count-prefix: seckill-buy-user-count # 請求用戶數量的限制頭標識
    buy-prod-stock-count: 100 # 初始化腔骨商品的庫存數量,存入redis,實際開發中,沒有此配置(初始化商品庫存,在洗頭膏添加搶購商品是)
    buy-prod-stock-prefix: seckill-buy-prod-stock # 搶購商品數量 頭標識
    buy-user-lock-prefix: seckill-buy-user-lock # 鎖定搶購用戶的頭標識
    buy-prod-lock-prefix: seckill-buy-prod-lock # 鎖定商品庫存鎖頭標識

3.2.2 SeckillConfig 配置類

點擊查看代碼
/**
 * Created On : 9/11/2022.
 * <p>
 * Author : huayu
 * <p>
 * Description: SeckillConfig
 */
@Data
@Component
@ConfigurationProperties(prefix = "kh96.seckill")
public class SeckillConfig {

    /*
        請求用戶數量的限制頭標識
     */
    private String  buyUserCountPrefix;

    /*
        初始化搶購商品的庫存數量
     */
    private Integer buyProdStockCount;

    /*
        搶購商品數量 頭標識
     */
    private String buyProdStockPrefix;

    /*
        鎖定搶購用戶的頭標識
     */
    private String buyUserLockPrefix;

    /*
        鎖定商品庫存鎖頭標識
     */
    private String buyProdLockPrefix;

}

3.2.3 搶購介面和實作類

3.2.3.1 介面
點擊查看代碼
/**
 * Created On : 9/11/2022.
 * <p>
 * Author : huayu
 * <p>
 * Description: 秒殺搶購的業務介面
 */
public interface SeckillService {


    /**
     * @author : huayu
     * @date   : 9/11/2022
     * @param  : [prodId, stockCount]
     * @return : void
     * @description : 初始化商品庫存到快取
     */
    boolean  initProdStock2Redis(String prodId,Integer stockCount);

    /**
     * @author : huayu
     * @date   : 9/11/2022
     * @param  : [productId]
     * @return : boolean
     * @description : 校驗搶購商品的請求數量是否達到上限
     */
    boolean checkBuyUserCount(String productId);


    /**
     * @author : huayu
     * @date   : 9/11/2022
     * @param  : [prodId, buyCount]
     * @return : boolean
     * @description : 校驗商品庫存是否充足
     */
    boolean checkProdStockEnough(String prodId,Integer buyCount);

    /**
     * @author : huayu
     * @date   : 9/11/2022
     * @param  : [userId, prodId]
     * @return : boolean
     * @description : 校驗用戶是否已經搶購過當前商品
     */
    boolean checkBuyUserBought(String userId,String prodId);


    /**
     * @author : huayu
     * @date   : 9/11/2022
     * @param  : [userId]
     * @return : boolean
     * @description : 校驗商品庫存是否鎖定
     */
    boolean checkProdStockLocked(String prodId);

    /**
     * @author : huayu
     * @date   : 9/11/2022
     * @param  : [prodId]
     * @return : boolean
     * @description : 釋放庫存鎖
     */
    void unLockProdStock(String prodId);

    /**
     * @author : huayu
     * @date   : 9/11/2022
     * @param  : [prodId, butCount]
     * @return : void
     * @description : 扣減庫存
     */
    void subProdStockCount(String prodId, Integer butCount);


}
3.2.3.2 實作類
點擊查看代碼
/**
 * Created On : 9/11/2022.
 * <p>
 * Author : huayu
 * <p>
 * Description: 搶購業務介面實作類
 */
@Service
public class SeckillServiceImpl implements SeckillService {

    @Autowired
    private RedisUtils redisUtils;

    @Autowired
    private SeckillConfig seckillConfig;


    @Override
    public boolean initProdStock2Redis(String prodId, Integer stockCount) {
        // 判斷redis中,是否存在已經初始化的商品庫存資料,如果已經存在,不需要再次初始化
        if (ObjectUtils.isEmpty(redisUtils.get(seckillConfig.getBuyProdStockPrefix() + ":" + prodId))){

            // 將商品庫存添加到redis中
            return redisUtils.set(seckillConfig.getBuyProdStockPrefix() + ":" + prodId, stockCount);
        }

        // 已經存在商品庫存,不需要設定
        return false;
    }


    @Override
    public boolean checkBuyUserCount(String prodId) {
        // 使用redis的incr操作,校驗當前商品的搶購用戶數是否達到上限
        return redisUtils.incr(seckillConfig.getBuyUserCountPrefix() + ":" + prodId, 1)
                > Integer.parseInt(redisUtils.get(seckillConfig.getBuyProdStockPrefix() + ":" + prodId).toString()) * 2;
    }


    @Override
    public boolean checkProdStockEnough(String prodId, Integer buyCount) {

        //校驗商品庫存,是否大于等于用戶搶購數量,如果小于,代表庫存不足
        //TODO 如果redis 沒有查到庫存(庫存沒初始化,或者后臺誤刪,必須要進行資料庫查詢操作,獲取庫存,
        // 要加鎖,只有一個執行緒取操作),再同步到redis

        return Integer.parseInt(redisUtils.get(seckillConfig.getBuyProdStockPrefix() + ":" + prodId).toString())
                >= buyCount;

    }

    @Override
    public boolean checkBuyUserBought(String userId, String prodId) {

        //使用redis的分布式鎖,sexnx,
        // 如果上鎖成功,代表沒有買過,可以繼續購買,如果上鎖失敗,表示購買過不能繼續購買
        //鎖時長為正搶購活動時長
        return ! redisUtils.lock( seckillConfig.getBuyUserLockPrefix()+":"+userId,
                                null,
                                15*60); //模擬 搶購時長
    }

    @Override
    public boolean checkProdStockLocked(String prodId) {
        //使用redis的分布式鎖,sexnx,如果上鎖成功,代表庫存沒有被鎖,如果上鎖失敗代表庫存被其他用戶鎖定不能購買
        //鎖庫存必須要增加時長限制,防止庫存鎖釋放失敗,導致用戶無法搶購,過期仍然會釋放
        return ! redisUtils.lock(seckillConfig.getBuyProdLockPrefix()+":"+prodId,
                                null,
                                2*60); //模擬鎖2分鐘

    }

    @Override
    public void unLockProdStock(String prodId) {
        //洗掉redis中的 庫存鎖
        redisUtils.del(seckillConfig.getBuyProdLockPrefix()+":"+prodId);
    }

    @Override
    public void subProdStockCount(String prodId, Integer butCount) {
        //扣減 redis中快取的商品庫存數量
        //TODO redis快取中操成功后,要發異步訊息 到佇列  更新資料庫中商品庫存呢
        redisUtils.decr(seckillConfig.getBuyProdStockPrefix() + ":" + prodId,
                butCount);

    }

}

3.2.4 遠程呼叫訂單模塊進行下單

點擊查看代碼
/**
 * Created On : 10/11/2022.
 * <p>
 * Author : huayu
 * <p>
 * Description: SeckillOrderFeignService
 */
@FeignClient(value = "https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/edocmall-seckill-order")
public interface SeckillOrderFeignService {


    /**
     * @author : huayu
     * @date   : 10/11/2022
     * @param  : [userId, prodId, buyCount]
     * @return : java.lang.String
     * @description : 遠程呼叫訂單中心,生成秒殺搶購訂單的介面
     */
    @GetMapping("/createSeckillOrder")
    String feignInvokeCreateSeckillOrder(@RequestParam(value = "https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/userId") String userId,
                              @RequestParam(value = "https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/prodId") String prodId,
                              @RequestParam(value = "https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/buyCount") Integer buyCount);

   

}

3.2.5 搶購 控制層

點擊查看代碼
/**
 * Created On : 9/11/2022.
 * <p>
 * Author : huayu
 * <p>
 * Description: 秒殺操作入口
 */
@Slf4j
@RestController
@Api(tags = "秒殺操作")
public class SeckillController {


    @Autowired
    private SeckillService seckillService;

    @Autowired
    private SeckillOrderFeignService seckillOrderFeignService;

    /**
     * @author : zhukang
     * @date   : 2022/11/9
     * @param  : [userId, prodId, buyCount]
     * @return : com.kgc.scd.util.RequestResult<?>
     * @description : 模擬初始化商品庫存,實際開發中,沒有此操作,是在后臺添加搶購商品預熱是,直接同步到redis
     */
    @GetMapping("/initProdStock")
    @ApiOperation(value = "https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/1 初始庫存", notes = "搶購商品預熱,初始化商品庫存數量到redis中")
    @ApiImplicitParams({
            @ApiImplicitParam(value = "https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/初始化商品id",name = "prodId"),
            @ApiImplicitParam(value = "https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/初始化商品數量",name = "stockCount")
    })
    public RequestResult<?> initProdStock(@RequestParam String prodId,
                                          @RequestParam Integer stockCount){

        log.info("------ 初始化商品:{},庫存數量:{},到redis快取中 ------", prodId, stockCount);

        // 增加業務邏輯處理:防止瞬時請求過多,使用redis的incr原子操作,進行請求用戶數的統計計數,如果請求已經達到了上限(比如:設定請求用戶數最多是搶購商品庫存的2倍),其它后續所有的請求都默認無效,回傳失敗:當前請求用戶過多,請稍后重試
        if(seckillService.initProdStock2Redis(prodId, stockCount)){
            return ResultBuildUtil.success("初始化商品庫存成功!");
        }
        // 初始化商品庫存失敗
        return ResultBuildUtil.fail("601", "初始化商品庫存失敗,請確認redis庫存是否已初始化!");
    }


    @GetMapping("/seckillBuy")
    @ApiOperation(value = "https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/2 開始秒殺", notes = "基于redis,RabbitMQ訊息佇列,實作有序秒殺搶購功能")
    @ApiImplicitParams({
        @ApiImplicitParam(value = "https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/用戶id",name = "userId"),
        @ApiImplicitParam(value = "https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/商品id",name = "prodId"),
        @ApiImplicitParam(value = "https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/搶購數量",name = "buyCount")
    })
    public RequestResult<?> seckillBuy(@RequestParam String userId,
                                       @RequestParam String prodId,
                                       @RequestParam Integer buyCount){

        log.info("------ 用戶:{},購買商品:{},購買數量:{},開始搶購 ------", userId, prodId, buyCount);

        // TODO 增加請求許可校驗(自定義注解+攔截器),校驗請求此介面的所有用戶是否是已登錄用戶,如果沒有登錄,提示請登錄
        // TODO 增加介面引數的校驗,判斷用戶是否是系統正常用戶,不能是狀態例外用戶,判斷商品的資料是否正確(切記:涉及系統內資料不要信任請求引數,要信任系統的快取的資料庫)

        // TODO 為了提高搶購入口的并發處理能力,要減少資料庫互動,可以設計為根據商品編號,從redis快取中查詢商品,如果商品資訊存在,則參與搶購,如果不存在,還是需要到資料庫查詢商品,如果資料庫中存在,將商品資訊存入redis快取,如果資料庫不存在,則直接提示搶購失敗,
        // TODO 此種場景,正常情況,沒有問題,可能存在的問題,某個商品,是首次參與搶購,快取中沒有資料,但是資料庫有,雖然上面的處理方式,可以解決,但是在高并發場景下,同一時刻會有大批量的請求來秒殺此商品,此時同時去快取中獲取商品資料,沒有獲取到,又同時去資料庫查詢,就會導致資料庫扛不住壓力,可能直接資料庫掛掉,
        // TODO 解決方式:快取商品資料一般都是在后臺添加搶購商品時,直接對商品進行預熱處理,即:事先把參與搶購的商品直接同步到redis快取中,這樣當搶購開始,直接從redis快取就可以獲取到商品,而不會出現快取擊穿問題,
        // TODO 雖然商品資料預熱方式,可以解決此類問題,但是可能還會存在例外(比如:快取中的商品由于后臺失誤操作,導致設定的過期時間不對,快取時間提前過期,或者快取資料誤洗掉),此種情況還是需要當快取不存在商品資料,從資料庫查詢,放入快取的邏輯,
        // TODO 解決方式:可以進行加鎖,限制在高并發的情況下訪問資料庫,如果同一時刻,快取中沒有獲取到商品資料庫,就進入查詢資料庫操作,但是在進入查詢前,增加分布式鎖,只有獲取到鎖的請求,才可以查詢資料庫并加入到快取中(加完就釋放鎖),獲取不到鎖的請求,直接回傳失敗(損失當前請求,后續請求進行彌補,只要一個操作成功,快取中的資料就存在,后續的請求自然可以獲取到資料)
        // TODO 極端情況:redis無法使用,必須要增加redis的高可用,確保redis永遠是有效的,考慮的模式就是集群模式下的哨兵機制,或者把大批量的請求直接放到訊息佇列,進行緩沖處理,

        log.info("------------------------------ 進行請求用戶數的統計計數,防止請求過多  -----------------------------------------");
        // 增加業務邏輯處理:防止瞬時請求過多,使用redis的incr原子操作,進行請求用戶數的統計計數,如果請求已經達到了上限(比如:設定請求用戶數最多是搶購商品庫存的2倍),其它后續所有的請求都默認無效,回傳失敗:當前請求用戶過多,請稍后重試
        if(seckillService.checkBuyUserCount(prodId)){
            return ResultBuildUtil.fail("602", "搶購失敗,當前搶購用戶過多,請稍后重試!");
        }

        log.info("------------------------------            查看庫存是否充足           -----------------------------------------");
        //校驗商品庫存數量是否充足,可以進行后續搶購,日過不足,直接搶購失敗
        log.info("------ 用戶:{},購買商品:{},購買數量:{},庫存校驗 ------", userId, prodId, buyCount);
        if(!seckillService.checkProdStockEnough(prodId,buyCount)){
            //庫存不足,回傳搶購失敗
            log.info("------ 用戶:{},購買商品:{},購買數量:{},庫存不足 ------", userId, prodId, buyCount);
            return ResultBuildUtil.fail("603", "搶購失敗,庫存不足,缺貨!");
        }

        log.info("------------------------------            判斷用戶是否搶購過           -----------------------------------------");
        //增加冪等操作:當前搶購用戶只能搶購一次,如果已經搶購過商品,不允許再次搶購(限制一個用戶同一個搶購商品,整個搶購期間只能搶購一次)
        log.info("------ 用戶:{},購買商品:{},購買數量:{},鎖定搶購用戶,如果 已經搶購過商品,不允許再次搶購 ------", userId, prodId, buyCount);
        if(seckillService.checkBuyUserBought(userId,prodId)){
            //用戶搶購過,回傳搶購失敗
            log.info("------ 用戶:{},購買商品:{},購買數量:{},重復搶購 ------", userId, prodId, buyCount);
            return ResultBuildUtil.fail("604", "搶購失敗,重復搶購!");
        }

        log.info("------------------------------            用戶獲取庫存鎖           -----------------------------------------");
        //執行搶購業務,先給商品庫存上鎖(鎖庫存),如果上鎖成功,代表當前用戶繼續搶購商品,如果上鎖失敗,說明有人搶購,進行等待
        log.info("------ 用戶:{},購買商品:{},購買數量:{},(嘗試獲取庫存鎖) 執行搶購業務,先給商品庫存上鎖(鎖庫存),拿到 庫存鎖 才可以開始下單 ------", userId, prodId, buyCount);
        while (true){
            //死回圈,嘗試鎖庫存,如果鎖庫存成功,代表庫存所已經被釋放
            if(!seckillService.checkProdStockLocked(prodId)){
                log.info("------ 用戶:{},購買商品:{},購買數量:{},鎖庫存成功,拿到 庫存鎖 ,開始下單------", userId, prodId, buyCount);
                //結束回圈,執行搶購下單
                break;
            }
            log.debug("------ 用戶:{},購買商品:{},購買數量:{},鎖庫存成功失敗,等待,,,------", userId, prodId, buyCount);
        }

        log.info("------------------------------     已經獲得庫存鎖,再次判斷庫存是否充足  -----------------------------------------");
        //考慮高并發的場景:
        //多人同時校驗庫存成功,但是進入到搶購下單業務時,庫存呢只夠部分人購買,需要再確定庫存是否足夠
        //校驗商品庫存數量是否充足,可以進行后續搶購,日過不足,直接搶購失敗
        log.info("------ 用戶:{},購買商品:{},購買數量:{},鎖庫存后(拿到庫存鎖后)  再次 庫存校驗 ------",userId, prodId, buyCount);
        if(!seckillService.checkProdStockEnough(prodId,buyCount)){
            //釋放庫存鎖,后續用戶繼續嘗試購買
            //庫存剩余2兩,還有三個人買,庫存不足,后續兩個人各買一件
            //釋放庫存鎖
            seckillService.unLockProdStock(prodId);
            log.info("------ 用戶:{},購買商品:{},購買數量:{},再次 庫存校驗,庫存不足 ------", userId, prodId, buyCount);

            //庫存不足,回傳搶購失敗
            log.info("------ 用戶:{},購買商品:{},購買數量:{},搶購下單中,庫存不足 ------", userId, prodId, buyCount);
            return ResultBuildUtil.fail("605", "搶購失敗,庫存不足,缺貨!");
        }

        log.info("------------------------------            開始扣減庫存           -----------------------------------------");
        //執行扣減商品庫存
        log.info("------ 用戶:{},購買商品:{},購買數量:{},扣減庫存 ------", userId, prodId, buyCount);
        seckillService.subProdStockCount(prodId,buyCount);


        log.info("------------------------------               開始下單           -----------------------------------------");
        //開始呼叫訂單中心的生成搶購訂單介面,下單并回傳給前端,搶購結果
        //先模擬下單成功,回傳一個搶購訂單號
        log.info("------ 用戶:{},購買商品:{},購買數量:{},開始下單 ------", userId, prodId, buyCount);
//        String seckillOrderNo = "SK"+ LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyyMMddHHmmss"))
//                + UUID.randomUUID().toString().substring(0,5);

        String seckillOrder = seckillOrderFeignService.feignInvokeCreateSeckillOrder(userId, prodId, buyCount);

        log.info("------------------------------       下單成功 主動釋放庫存鎖        -----------------------------------------");
        //生成搶購訂單成功,立刻釋放庫存鎖,給其他搶購用戶購買
        log.info("------ 用戶:{},購買商品:{},購買數量:{},下單成功,釋放庫存鎖 ------", userId, prodId, buyCount);
        seckillService.unLockProdStock(prodId);

        log.info("------ 用戶:{},購買商品:{},購買數量:{},搶購成功 ------", userId, prodId, buyCount);

        //回傳搶購成功,實際開發不能回傳此種格式的資料
        //必須使用key和value的回傳,方便前端獲取訂單號
        return ResultBuildUtil.success("搶購成功,搶購訂單"+seckillOrder);
    }



}

3.3 訂單模塊

3.3.1 application.yml

點擊查看代碼
# 埠
server:
  port: 8107

# 服務名
spring:
  application:
    name: edocmall-seckill-order

  # redis
  redis:
    host: 127.0.0.1
    port: 6379

  # RabbitMQ
  rabbitmq:
    host: 1.117.75.57
    port: 5672
    username: admin
    password: admin

# eureka 注冊中心的配置
eureka:
  client:
    service-url:
      defaultZone: http://127.0.0.1:8096/eureka  # 注冊中心的地址

3.3.2 OrderMQDirectConfig

點擊查看代碼
/**
 * Created On : 1/11/2022.
 * <p>
 * Author : huayu
 * <p>
 * Description: Direct直連模式,自動配置類,自動創建佇列,交換機,并將佇列系結到交換機,指定唯一路由
 */
@Configuration
public class OrderMQDirectConfig {

    /**
     * @author : huayu
     * @date   : 1/11/2022
     * @param  : []
     * @return : org.springframework.amqp.core.Queue
     * @description : directQueue
     */
    @Bean
    public Queue directQueue(){

        return new Queue(OrderMQConstant.SECKILL_SAVE_ORDER_QUEUE_KH96,true);
    }

    /**
     * @author : huayu
     * @date   : 1/11/2022
     * @param  : []
     * @return : org.springframework.amqp.core.DirectExchange
     * @description : 創建直連交換機
     */
    @Bean
    public DirectExchange directExchange(){
        // 創建支持持久化的直連交換機,指定交換機的名稱
        return new DirectExchange(OrderMQConstant.SECKILL_SAVE_ORDER_EXCHANGE_KH96U);
    }

    /**
     * @author : huayu
     * @date   : 1/11/2022
     * @param  : []
     * @return : org.springframework.amqp.core.Binding
     * @description : 將直連佇列和直連交換機進行系結,并指定系結的唯一路由鍵
     */
    @Bean
    public Binding directBinding(){
        // 將直連佇列和直連交換機進行系結,并指定系結的唯一路由鍵
        return BindingBuilder.bind(directQueue())
                            .to(directExchange())
                            .with(OrderMQConstant.SECKILL_SAVE_ORDER_ROUTING_KH96);
    }


}

3.3.3 OrderMQConstant

點擊查看代碼
/**
 * Created On : 1/11/2022.
 * <p>
 * Author : huayu
 * <p>
 * Description: RabbitMQ 常量類,系統的所有佇列名,交換機名,路由鍵名等,統一進行配置管理
 */
public class OrderMQConstant {

    //========================== 直連模式
    /**
     * Direct直連模式 佇列名
     */
    public static final String SECKILL_SAVE_ORDER_QUEUE_KH96 ="seckill_save_order_queue_kh96";

    /**
     * Direct直連模式 交換機名
     */
    public static final String SECKILL_SAVE_ORDER_EXCHANGE_KH96U ="seckill_save_order_exchange_kh96";

    /**
     * Direct直連模式 路由鍵
     */
    public static final String SECKILL_SAVE_ORDER_ROUTING_KH96 ="seckill_save_order_routing_kh96";


    //========================== 扇形模式

    /**
     * Fanout 扇形模式 佇列名
     */
    public static final String ACCOUNT_FANOUT_QUEUE_KH96 ="account_pay_result_queue_kh96";

}

3.3.4 下訂單業務層

3.3.4.1 介面
點擊查看代碼
/**
 * Created On : 10/11/2022.
 * <p>
 * Author : huayu
 * <p>
 * Description:
 */
public interface SeckillOrderService {

    /**
     * @author : huayu
     * @date   : 11/11/2022
     * @param  : [seckillOrder]
     * @return : void
     * @description : 生成秒殺訂單
     */
    void saveSeckillOrder(Map<String,Object> seckillOrder);

}
3.3.4.2實作類
點擊查看代碼
/**
 * Created On : 10/11/2022.
 * <p>
 * Author : huayu
 * <p>
 * Description: SeckillOrderServiceImpl
 */
@Service
@Slf4j
public class SeckillOrderServiceImpl implements SeckillOrderService {

    @Autowired
    private RabbitTemplate rabbitTemplate;

    @Autowired
    private RedisUtils redisUtils;


    @Override
    public void saveSeckillOrder(Map<String, Object> seckillOrder) {

        //發送生成搶購訂單的訊息到訊息佇列,并在redis中添加此訂單的記錄,模擬互動
        //0 正在生成
        if(redisUtils.set(seckillOrder.get("seckillOrderNo").toString(),0)){
            //發送生成訂單的訊息
            rabbitTemplate.convertAndSend(OrderMQConstant.SECKILL_SAVE_ORDER_EXCHANGE_KH96U,
                    OrderMQConstant.SECKILL_SAVE_ORDER_ROUTING_KH96,
                    seckillOrder);

        }

    }


}

3.3.5 SeckillOrderSaveListener

點擊查看代碼
/**
 * Created On : 1/11/2022.
 * <p>
 * Author : huayu
 * <p>
 * Description: Direct 直連模式消費者 One
 */
@Slf4j
@Component
//指定接聽的 訊息佇列 名字
@RabbitListener(queues = OrderMQConstant.SECKILL_SAVE_ORDER_QUEUE_KH96)
public class SeckillOrderSaveListener {

    @Autowired
    private RedisUtils redisUtils;

    /**
     * @author : huayu
     * @date   : 1/11/2022
     * @param  : [directMsgJson]
     * @return : void
     * @description : Direct 直連模式消費者One,消費資訊
     */
    //指定訊息佇列中的訊息,交給對應的方法處理
    @RabbitHandler
    public void saveSeckillOrder(Map<String,Object> seckillOrder){
        log.info("***** 秒殺搶購訂單:{},開始入庫 ******",seckillOrder.get("seckillOrderNo"));

        //TODO 將訊息中的訂單物體物件,調入業務介面,插入到資料庫

        redisUtils.set(seckillOrder.get("seckillOrderNo").toString(),1);

        log.info("***** 秒殺搶購訂單:{},入庫成功 ******",seckillOrder.get("seckillOrderNo"));


    }


}

3.3.6 下單 控制層

點擊查看代碼
/**
 * Created On : 10/11/2022.
 * <p>
 * Author : huayu
 * <p>
 * Description: 秒殺搶購訂單入口
 */
@Slf4j
@RestController
public class SeckillOrderController {

    @Autowired
    private SeckillOrderService seckillOrderService;


    /**
     * @author : huayu
     * @date   : 10/11/2022
     * @param  : [userId, prodId, buyCount]
     * @return : java.lang.String
     * @description : 生成秒殺搶購訂單
     */
    @GetMapping("/createSeckillOrder")
    public String createSeckillOrder(@RequestParam String userId,
                                     @RequestParam String prodId,
                                     @RequestParam Integer buyCount){

        log.info("****** 用戶:{},購買商品:{},購買數量:{},生成搶購訂單 ******", userId, prodId, buyCount);

        //TODO 必須要有引數校驗,必須要有用戶,商品資訊的校驗,確定用戶是否正常,商品是否還在搶購中
        //TODO 再次強調所有的中心模塊,資料來源,不能信任外部介面來源的引數,都必須從資料庫或者快取中獲取,尤其是跟金錢相關
        //TODO 所有的介面必需要校驗結束,通過獲取的資料,封裝訂單物體物件,用戶不關系訂單的生成業務,可以使用異步訊息佇列,實作曉峰,并快速回應

        //模擬生成一個搶購訂單號,并封裝成訂單物體物件,通過map集合模擬
        String seckillOrderNo = "SK"+ LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyyMMddHHmmss"))
                + UUID.randomUUID().toString().substring(0,5);

        //創建訂單物體物件
        Map<String, Object> seckillOrder = new HashMap<>();
        seckillOrder.put("userId",userId);
        seckillOrder.put("prodId",prodId);
        seckillOrder.put("buyCount",buyCount);

        //TODO 其他的訂單屬性封裝,比如識訓人,總價格,優惠,時間等
        seckillOrder.put("seckillOrderNo",seckillOrderNo);

        //發送到生成訂單的訊息佇列中,使用異步處理
        seckillOrderService.saveSeckillOrder(seckillOrder);

        //回傳搶購訂單
        return  seckillOrder.toString();

    }

}

3.4 查詢訂單狀態和訂單支付部分不再贅述

4、測驗

4.1 初始化庫存

4.2 搶購

4.2.1 搶購情況

4.2.2 redis中資料變化情況

4.3 查看訂單詳情

4.4 訂單支付

支付的時候數注意穿透的路徑;




轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/538348.html

標籤:其他

上一篇:Java中List同ArrayList有什么不同呢?

下一篇:基于sklearn的集成學習實戰

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more