SpringCloud(十一)- 秒殺 搶購

来源:https://www.cnblogs.com/xiaoqigui/archive/2022/11/25/16925343.html
-Advertisement-
Play Games

1、流程圖 1.1 數據預熱 1.2 搶購 1.3 生成訂單 (發送訂單消息) 1.4 訂單入庫 (監聽 消費訂單消息) 1.5 查看訂單狀態 1.6 支付 (獲取支付鏈接 ) 1.7 支付成功 微信回調 (發送 支付成功消息) 1.8 支付成功 返回給前端成功 (監聽 支付成功消息) 2、incr ...


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 = "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 = "userId") String userId,
                              @RequestParam(value = "prodId") String prodId,
                              @RequestParam(value = "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 = "1 初始庫存", notes = "搶購商品預熱,初始化商品庫存數量到redis中")
    @ApiImplicitParams({
            @ApiImplicitParam(value = "初始化商品id",name = "prodId"),
            @ApiImplicitParam(value = "初始化商品數量",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 = "2 開始秒殺", notes = "基於redis,RabbitMQ消息隊列,實現有序秒殺搶購功能")
    @ApiImplicitParams({
        @ApiImplicitParam(value = "用戶id",name = "userId"),
        @ApiImplicitParam(value = "商品id",name = "prodId"),
        @ApiImplicitParam(value = "搶購數量",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 訂單支付

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





您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一.小結 1.使用語法elemenrType[ ] arrayRefVar(元素類型[ ] 數組引用變數)或elementType arrayRefVar[ ](元素類型 數組引用變數[ ])聲明一個數組類型的變數。儘管elementType[ ] arrayaRefVar也是合法的,但是還是推薦使 ...
  • 多個線程在執行過程中會因為競爭同一個資源而產生線程衝突,造成死鎖,從而引出線程鎖這個概念 先拿到鎖再執行業務操作: 當然我對這一塊瞭解的還不透徹,只是瞭解在不加鎖的多線程情況下,會破壞單例模式,所以就有了下麵這一段 1 import time 2 import threading 3 4 5 def ...
  • ##java原生態中的匿名內部類 ###1.匿名內部類的定義 使用匿名內部類的兩種的方法 建立父類,重寫父類的方法 實現介面的方法 ###2.普通類的實現 ####1. 普通類實現 實現普通類需要先聲明對一個類的對象,再調用對象的方法 建立一個example類 public void student ...
  • MyQueue版本1 #include <iostream> using namespace std; template<typename T> class MyQueue { private: struct QueueItem { QueueItem(T _data = T(), QueueIte ...
  • 1、登錄 1.1 登錄的時候做vip的判斷; 1.2 使用JWT(Java Web token),驗證登錄,更加安全 2、連續簽到 2.1判斷是否斷簽: ​ 通過判斷昨天是否登錄,可以判斷; 2.2判斷連續簽到多少天: ​ 將每次簽到的記錄保存在redis中,判斷保存的數量,有多少個,就連續簽到多少 ...
  • 1、使用質數定義計算 #version1import datetime #導入模塊計算效率start = datetime.datetime.now() count = 0 for x in range(2,100000): #求指定範圍內的質數 for i in range(2,x): #除以1和 ...
  • Spring Boot 3.0 正式發佈 大家好,我是棧長。 Spring Boot 3.0 正式發佈了: 同時發佈更新的還有 2.7.x 和 2.6.x 兩條版本線,Spring Boot 是我見過的發版最守時的技術框架之一。 Spring Boot 3.0 這是一個重大的主版本更新,距離上一代的 ...
  • 本文主要跟隨Datawhale的學習路線以及內容教程,詳細介紹了集成學習常見的多個演算法的基於sklearn的實現過程,同時還有兩個案例,內容豐富。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...