限量搶購功能的實現

業務背景:

        某公司開展活動,4個實體店,每天限量發放10個優惠券,供人們搶購。

技術分析:

        一旦涉及搶購,必然會引起併發問題,而且還儘量保證程序的併發性,這就要求或者在程序中做同步,或者在數據庫層次做同步。一般來說,在程序端做同步(synchronize),會導致阻塞得很厲害。使用數據庫同步,可以選擇有行級鎖的數據庫以及提高事務隔離等級,但提高事務等級會將事務串行化(Serializable),嚴重降低程序併發。綜上權衡,我選擇了PHP+MYSQL。以我的認知,Oracle與MySQL都可以,MSSQL卻沒找到類似功能。

主要解決方法:

        1.需要把實體店對應的優惠券數量以記錄的形式呈現,表1:Max_Table( ID, storeID,apply_count ,apply_code)。表2:apply_form(apply_code,store_id,phone_numbe)該表主要記錄哪個用戶用什麼手機號碼申請了優惠券,優惠券碼是什麼。

        2.創建數據庫時要使用InnoDB存儲引擎

        3.在數據庫中創建存儲過程,在存儲過程中使用事務,在事務中要先查詢是否發放完畢,爲避免多進程同時訪問某記錄中的數量而引起的問題,我使用for update。使用該語句,若事務A訪問訪問該記錄時,事務B會阻塞,直到事務A完畢後,從而達到了<並轉串>的操作。

以下是解決該問題的核心存儲過程,其中有2個入參,1個出參。

CREATE DEFINER=`sa`@`localhost` PROCEDURE `p_apply_code`(IN `p_store_id` varchar(30), IN `p_phone_number` varchar(20),OUT  return_val varchar(30))  
BEGIN  

    #Routine body goes here...  

  declare p_apply_count int(4) default 0;  

  declare encode_apply_code varchar(16) default '';  

  start transaction ;//開始事務  

  SELECT apply_count into p_apply_count FROM max_table WHERE store_id=p_store_id   for update;  

  if (p_apply_count =0 ) THEN  

  insert into max_table(store_id,apply_count) values(p_store_id,0);  

  end if;  

 IF( p_apply_count<=9) THEN   

set encode_apply_code='優惠券碼的生成規則',  

 insert into apply_form(apply_code,store_id,phone_number) values(encode_apply_code,p_store_id,p_phone_number);  

 update max_table set apply_count=apply_count+1 where store_id=p_store_id;  

set result=encode_apply_code2;    

  END IF;  

 set return_val=result ;  

 commit; //提交事務,同時釋放for update鎖    

END  


    

發佈了1 篇原創文章 · 獲贊 0 · 訪問量 3萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章