自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Java秒殺系統(tǒng):商品秒殺代碼實戰(zhàn)

新聞 前端
“商品秒殺”功能模塊是建立在“商品詳情”功能模塊的基礎(chǔ)之上,對于這一功能模塊而言,其主要的核心流程在于:前端發(fā)起搶購請求,該請求將攜帶著一些請求數(shù)據(jù)。

內(nèi)容:

“商品秒殺”功能模塊是建立在“商品詳情”功能模塊的基礎(chǔ)之上,對于這一功能模塊而言,其主要的核心流程在于:前端發(fā)起搶購請求,該請求將攜帶著一些請求數(shù)據(jù):待秒殺Id跟當(dāng)前用戶Id等數(shù)據(jù);后端接口在接收到請求之后,將執(zhí)行一系列的判斷與秒殺處理邏輯,最終將處理結(jié)果返回給到前端。

[[271508]]

其中,后端接口的這一系列判斷與秒殺處理邏輯還是挺復(fù)雜的,Debug將其繪制成了如下的流程圖:

Java秒殺系統(tǒng)實戰(zhàn)系列:商品秒殺代碼實戰(zhàn)

從該業(yè)務(wù)流程圖中可以看出,后端接口在接收前端用戶的秒殺請求時,其核心處理邏輯為:

(1)首先判斷當(dāng)前用戶是否已經(jīng)搶購過該商品了,如果否,則代表用戶沒有搶購過該商品,可以進入下一步的處理邏輯

(2)判斷該商品可搶的剩余數(shù)量,即庫存是否充足(即是否大于0),如果是,則進入下一步的處理邏輯

(3)扣減庫存,并更新數(shù)據(jù)庫的中對應(yīng)搶購記錄的庫存(一般是減一操作),判斷更新庫存的數(shù)據(jù)庫操作是否成功了,如果是,則創(chuàng)建用戶秒殺成功的訂單,并異步發(fā)送短信或者郵件通知信息通知用戶

(4)以上的操作邏輯如果有任何一步是不滿足條件的,則直接結(jié)束整個秒殺的流程,即秒殺失?。?/p>

接下來,我們?nèi)匀换贛VC的開發(fā)模式,采用代碼實戰(zhàn)實現(xiàn)這一功能模塊!

(1)首先是在KillController 控制器開發(fā)接收“前端用戶秒殺請求”的功能方法,其中,該方法需要接收前端請求過來的“待秒殺Id”,而當(dāng)前用戶的Id可以通過上一篇博文介紹的Shiro 的會話模塊Session進行獲取!

其源代碼如下所示:

  1. private static final String prefix = "kill"
  2.   
  3. @Autowired 
  4. private IKillService killService; 
  5.   
  6. @Autowired 
  7. private ItemKillSuccessMapper itemKillSuccessMapper; 
  8.   
  9. /*** 
  10.  * 商品秒殺核心業(yè)務(wù)邏輯 
  11.  */ 
  12. @RequestMapping(value = prefix+"/execute",method = RequestMethod.POST,consumes = MediaType.APPLICATION_JSON_UTF8_VALUE) 
  13. @ResponseBody 
  14. public BaseResponse execute(@RequestBody @Validated KillDto dto, BindingResult result, HttpSession session){ 
  15.  if (result.hasErrors() || dto.getKillId()<=0){ 
  16.  return new BaseResponse(StatusCode.InvalidParams); 
  17. //獲取當(dāng)前登錄用戶的信息 
  18.  Object uId=session.getAttribute("uid"); 
  19.  if (uId==null){ 
  20.  return new BaseResponse(StatusCode.UserNotLogin); 
  21.  } 
  22.  Integer userId= (Integer)uId ; 
  23.  BaseResponse response=new BaseResponse(StatusCode.Success); 
  24.  try { 
  25.  Boolean res=killService.killItem(dto.getKillId(),userId); 
  26.  if (!res){ 
  27.  return new BaseResponse(StatusCode.Fail.getCode(),"哈哈~商品已搶購?fù)戤吇蛘卟辉趽屬彆r間段哦!"); 
  28.  } 
  29.  }catch (Exception e){ 
  30.  response=new BaseResponse(StatusCode.Fail.getCode(),e.getMessage()); 
  31.  } 
  32.  return response; 
  33. }復(fù)制代碼 

其中,KillDto對象主要封裝了“待秒殺Id”等字段信息,其主要用于接收前端過來的用戶秒殺請求信息,源代碼如下所示:

  1. @Data 
  2. @ToString 
  3. public class KillDto implements Serializable{ 
  4.  @NotNull 
  5.  private Integer killId; 
  6.   
  7.  private Integer userId; //在整合shiro之后,userId字段可以不需要了!因為通過session進行獲取了 
  8. }復(fù)制代碼 

(2)緊接著是開發(fā) killService.killItem(dto.getKillId(),userId) 的功能,該功能對應(yīng)的代碼的編寫邏輯可以參見本文剛開始介紹時的流程圖!其完整源代碼如下所示:

  1. @Autowired 
  2. private ItemKillSuccessMapper itemKillSuccessMapper; 
  3.   
  4. @Autowired 
  5. private ItemKillMapper itemKillMapper; 
  6.   
  7. @Autowired 
  8. private RabbitSenderService rabbitSenderService; 
  9.   
  10. //商品秒殺核心業(yè)務(wù)邏輯的處理 
  11. @Override 
  12. public Boolean killItem(Integer killId, Integer userId) throws Exception { 
  13.  Boolean result=false
  14.   
  15.  //TODO:判斷當(dāng)前用戶是否已經(jīng)搶購過當(dāng)前商品 
  16.  if (itemKillSuccessMapper.countByKillUserId(killId,userId) <= 0){ 
  17.  //TODO:查詢待秒殺商品詳情 
  18.  ItemKill itemKill=itemKillMapper.selectById(killId); 
  19.   
  20.  //TODO:判斷是否可以被秒殺canKill=1? 
  21.  if (itemKill!=null && 1==itemKill.getCanKill() ){ 
  22.  //TODO:扣減庫存-減一 
  23.  int res=itemKillMapper.updateKillItem(killId); 
  24.   
  25.  //TODO:扣減是否成功?是-生成秒殺成功的訂單,同時通知用戶秒殺成功的消息 
  26.  if (res>0){ 
  27.  commonRecordKillSuccessInfo(itemKill,userId); 
  28.   
  29.  result=true
  30.  } 
  31.  } 
  32.  }else
  33.  throw new Exception("您已經(jīng)搶購過該商品了!"); 
  34.  } 
  35.  return result; 
  36. }復(fù)制代碼 

其中,itemKillMapper.selectById(killId); 表示用于獲取待秒殺商品的詳情信息,這在前面的篇章中已經(jīng)介紹過了;而 itemKillMapper.updateKillItem(killId); 主要用于扣減庫存(在這里是減1操作),其對應(yīng)的動態(tài)Sql如下所示:

  1. <!--搶購商品,剩余數(shù)量減一--> 
  2.  <update id="updateKillItem"
  3.  UPDATE item_kill 
  4.  SET total = total - 1 
  5.  WHERE 
  6.  id = #{killId} 
  7.  </update>復(fù)制代碼 

(3)值得一提的是,在上面 KillService執(zhí)行killItem功能方法時,還開發(fā)了一個通用的方法:用戶秒殺成功后創(chuàng)建秒殺訂單、并異步發(fā)送通知消息給到用戶秒殺成功的信息!該方法為 commonRecordKillSuccessInfo(itemKill,userId); 其完整的源代碼如下所示:

  1. /** 
  2.  * 通用的方法-用戶秒殺成功后創(chuàng)建訂單-并進行異步郵件消息的通知 
  3.  * @param kill 
  4.  * @param userId 
  5.  * @throws Exception 
  6.  */ 
  7. private void commonRecordKillSuccessInfo(ItemKill kill, Integer userId) throws Exception{ 
  8.  //TODO:記錄搶購成功后生成的秒殺訂單記錄 
  9.   
  10.  ItemKillSuccess entity=new ItemKillSuccess(); 
  11.  String orderNo=String.valueOf(snowFlake.nextId()); 
  12.   
  13.  //entity.setCode(RandomUtil.generateOrderCode()); //傳統(tǒng)時間戳+N位隨機數(shù) 
  14.  entity.setCode(orderNo); //雪花算法 
  15.  entity.setItemId(kill.getItemId()); 
  16.  entity.setKillId(kill.getId()); 
  17.  entity.setUserId(userId.toString()); 
  18.  entity.setStatus(SysConstant.OrderStatus.SuccessNotPayed.getCode().byteValue()); 
  19.  entity.setCreateTime(DateTime.now().toDate()); 
  20.  //TODO:學(xué)以致用,舉一反三 -> 仿照單例模式的雙重檢驗鎖寫法 
  21.  if (itemKillSuccessMapper.countByKillUserId(kill.getId(),userId) <= 0){ 
  22.  int res=itemKillSuccessMapper.insertSelective(entity); 
  23.   
  24.  if (res>0){ 
  25.  //TODO:進行異步郵件消息的通知=rabbitmq+mail 
  26.  rabbitSenderService.sendKillSuccessEmailMsg(orderNo); 
  27.   
  28.  //TODO:入死信隊列,用于 “失效” 超過指定的TTL時間時仍然未支付的訂單 
  29.  rabbitSenderService.sendKillSuccessOrderExpireMsg(orderNo); 
  30.  } 
  31.  } 
  32. }復(fù)制代碼 

該方法涉及的功能模塊稍微比較多,即主要包含了“分布式唯一ID-雪花算法的應(yīng)用”、“整合RabbitMQ異步發(fā)送通知消息給用戶”、“基于JavaMail開發(fā)發(fā)送郵件的功能”、“死信隊列失效超時未支付的訂單”等等,這些功能模塊將在后面的小節(jié)一步一步展開進行介紹!

(4)最后是需要在前端頁面info.jsp開發(fā)“提交用戶秒殺請求”的功能,其部分核心源代碼如下所示:

Java秒殺系統(tǒng)實戰(zhàn)系列:商品秒殺代碼實戰(zhàn)

其中,提交的數(shù)據(jù)是采用application/json的格式提交的,即json的格式!并采用POST的請求方法進行交互!

(5)將整個系統(tǒng)、項目采用外置的tomcat運行起來,觀察控制臺的輸出信息,如果沒有報錯信息,則代表整體的實戰(zhàn)代碼沒有語法級別的錯誤!點擊“詳情”按鈕,登錄成功后,進入“待秒殺商品的的詳情”,可以查看當(dāng)前待秒殺商品的詳情信息;點擊“搶購”按鈕,即可進入“秒殺”環(huán)節(jié),后端經(jīng)過一系列的邏輯處理之后,將處理的結(jié)果返回給到前端,如下圖所示:

[[271509]]

與此同時,當(dāng)前用戶的郵箱中將收到一條“秒殺成功”的郵件信息,表示當(dāng)前用戶已經(jīng)成功秒殺搶到當(dāng)前商品了,如下圖所示:

Java秒殺系統(tǒng)實戰(zhàn)系列:商品秒殺代碼實戰(zhàn)

除此之外,在數(shù)據(jù)庫表item_kill_success中也將會生成一筆“秒殺成功的訂單記錄”,如下圖所示:

Java秒殺系統(tǒng)實戰(zhàn)系列:商品秒殺代碼實戰(zhàn)

當(dāng)然,對于“郵件的通知”和“秒殺成功生成的訂單的訂單編號”的功能,在本節(jié)我們主要是分享介紹了秒殺系統(tǒng)中用戶的“秒殺/搶購請求”功能!

責(zé)任編輯:張燕妮 來源: 今日頭條
相關(guān)推薦

2020-09-01 07:47:32

Redis秒殺微信

2020-10-14 07:20:53

高并發(fā)

2016-01-06 10:10:25

2018-09-15 04:59:01

2021-07-09 07:21:40

SpringBootRedisLUA

2020-04-01 17:31:03

Redis系統(tǒng)秒殺

2023-11-27 18:07:05

Go并發(fā)編程

2020-12-02 06:14:35

秒殺系統(tǒng)

2022-08-26 10:24:48

架構(gòu)Golang

2024-11-25 09:10:03

2021-12-03 10:47:28

WOT技術(shù)峰會技術(shù)

2024-10-10 17:23:31

2021-12-20 07:03:54

秒殺系統(tǒng)擴容

2021-06-23 06:48:42

秒殺Java電商

2025-02-20 00:01:00

2025-03-27 01:10:00

HashMap分段鎖CAS

2019-12-19 10:10:45

秒殺系統(tǒng)高并發(fā)

2021-07-29 08:13:05

高并發(fā)秒殺商品秒殺系統(tǒng)

2018-06-24 08:40:21

秒殺架構(gòu)架構(gòu)優(yōu)化
點贊
收藏

51CTO技術(shù)棧公眾號