首頁  >  文章  >  後端開發  >  PHP透過加鎖實現並發搶購功能

PHP透過加鎖實現並發搶購功能

*文
*文原創
2017-12-29 18:09:042347瀏覽

本文基於php語言使用加鎖實作並發情況下搶購功能,特定時段開放搶購並不允許開放的碼重複。本文介紹的非常詳細,需要的朋友參考下。希望對大家有幫助。

需求:搶碼功能

要求:

#1、特定時間段才開放搶碼;




2.每個時間段放開的碼是有限的;

3、每個碼不允許重複;


##實作:


1、在不考慮並發的情況下實現:

<p style="margin-top: 5px; margin-bottom: 5px;">function get_code($len){<br/>$CHAR_ARR = array(&#39;1&#39;,&#39;2&#39;,&#39;3&#39;,&#39;4&#39;,&#39;5&#39;,&#39;6&#39;,&#39;7&#39;,&#39;8&#39;,&#39;9&#39;,&#39;A&#39;,&#39;B&#39;,&#39;C&#39;,&#39;D&#39;,&#39;E&#39;,&#39;F&#39;,&#39;G&#39;,&#39;H&#39;,&#39;I&#39;,&#39;J&#39;,&#39;K&#39;,&#39;L&#39;,&#39;M&#39;,&#39;N&#39;,&#39;O&#39;,&#39;P&#39;,&#39;Q&#39;,&#39;X&#39;,&#39;Y&#39;,&#39;Z&#39;,&#39;W&#39;,&#39;S&#39;,&#39;R&#39;,&#39;T&#39;);<br/>$CHAR_ARR_LEN = count($CHAR_ARR) - 1;<br/>$code = &#39;&#39;;<br/>while(--$len > 0){ $code .= $CHAR_ARR[rand(0,$CHAR_ARR_LEN)]; }<br/>return $code;<br/>}<br/>$pdo = new PDO(&#39;mysql:host=localhost;dbname=ci_test&#39;,&#39;root&#39;,&#39;root&#39;);<br/>//查询当前时间已发放验证码数量<br/>$code_num_rs = $pdo->query("SELECT COUNT(*) as sum FROM code_test");<br/>$code_num_arr = $code_num_rs->fetch(PDO::FETCH_ASSOC);<br/>$code_num = $code_num_arr[&#39;sum&#39;];<br/>if($code_num < 1){<br>   sleep(2); //暂停2秒<br/>$code = get_code(6);<br/>var_dump( $pdo->query("INSERT INTO code_test (code,create_time) VALUES (&#39;$code&#39;,".time().")") );<br/>}<br/></p>

  上述程式碼預設滿足目前是開放時間,和碼是不重複的; 

     在不考慮並發情況下流程:


  1)選查詢目前資料庫發放的驗證碼數量;


  2)如果還有名額,則產生驗證碼,插入到資料庫,返回驗證碼到客戶端;


  3)如果已滿;則返回提示,已無名額;

2、並發情況下實現:


################################################################################################# #  那麼看下上面程式碼在並發情況下得到的結果:#########  測試並發,可以使用apache benchmark來測試,apache benchmark是APACHE旗下的HTTP SERVER的性能評測工具,通過cmd進入到apche的bin目錄下,透過ab指令調用,如:ab -c 並發數量-n 總訪問量url###
<p style="margin-top: 5px; margin-bottom: 5px;">cb -c 100 -n 100 http://localhost/php_mulit.php<br/></p>
###  這樣就是100個用戶同事去搶1個名額,在查詢的時候,每個使用者都查詢到還有一個名額,則會去產生驗證碼,插入資料庫,回傳驗證碼;這樣就造成了驗證碼發多了。事實上,運行完該指令,資料庫多了13筆記錄,而不是一筆。 #########  怎麼避免這情況發生呢? ######     可以透過加排他鎖來鎖定判斷到插入這個過程,保證這個判斷流程任一時間只有一個進程在運作。實作如下:###
<p style="margin-top: 5px; margin-bottom: 5px;">//生成码<br/>function get_code($len){<br/>$CHAR_ARR = array(&#39;1&#39;,&#39;2&#39;,&#39;3&#39;,&#39;4&#39;,&#39;5&#39;,&#39;6&#39;,&#39;7&#39;,&#39;8&#39;,&#39;9&#39;,&#39;A&#39;,&#39;B&#39;,&#39;C&#39;,&#39;D&#39;,&#39;E&#39;,&#39;F&#39;,&#39;G&#39;,&#39;H&#39;,&#39;I&#39;,&#39;J&#39;,&#39;K&#39;,&#39;L&#39;,&#39;M&#39;,&#39;N&#39;,&#39;O&#39;,&#39;P&#39;,&#39;Q&#39;,&#39;X&#39;,&#39;Y&#39;,&#39;Z&#39;,&#39;W&#39;,&#39;S&#39;,&#39;R&#39;,&#39;T&#39;);<br/>$CHAR_ARR_LEN = count($CHAR_ARR) - 1;<br/>$code = &#39;&#39;;<br/>while(--$len > 0){ $code .= $CHAR_ARR[rand(0,$CHAR_ARR_LEN)]; }<br/>return $code;<br/>}<br/>$pdo = new PDO(&#39;mysql:host=localhost;dbname=ci_test&#39;,&#39;root&#39;,&#39;root&#39;);<br/>$fp = fopen(&#39;lock.txt&#39;,&#39;r&#39;);<br/>//通过排他锁 锁定该过程<br/>if(flock($fp,LOCK_EX)){<br/>//查询当前时间已发放验证码数量<br/>$code_num_rs = $pdo->query("SELECT COUNT(*) as sum FROM code_test");<br/>$code_num_arr = $code_num_rs->fetch(PDO::FETCH_ASSOC);<br/>$code_num = $code_num_arr[&#39;sum&#39;];<br/>if($code_num < 1){<br/>sleep(2);<br/>$code = get_code(6);<br/>var_dump( $pdo->query("INSERT INTO code_test (code,create_time) VALUES (&#39;$code&#39;,".time().")") );<br/>}<br/>flock($fp,LOCK_UN);<br/>fclose($fp);<br/>}<br/></p>
###透過flock函數來鎖定該過程。 ######再次執行 ###
<p style="margin-top: 5px; margin-bottom: 5px;">cb -c 100 -n 100 http://localhost/php_mulit.php<br/></p>
###資料庫只增加了一筆記錄,保證了並發情況下資料的正確。 ############相關推薦:###############瀏覽器退出之後php還會繼續執行嗎的深入探究######################################################### ###########php實作線上通訊錄的實例###################PHP如何有效率地讀取大檔案的實例對比#### #####

以上是PHP透過加鎖實現並發搶購功能的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn