摘要:在秒殺,搶購等并發(fā)場景下,可能會出現(xiàn)超賣的現(xiàn)象,在語言中并沒有原生提供并發(fā)的解決方案,因此就需要借助其他方式來實現(xiàn)并發(fā)控制。借助文件排他鎖,在處理下單請求的時候,用鎖定一個文件,成功拿到鎖的才能處理訂單。
在秒殺,搶購等并發(fā)場景下,可能會出現(xiàn)超賣的現(xiàn)象,在PHP語言中并沒有原生提供并發(fā)的解決方案,因此就需要借助其他方式來實現(xiàn)并發(fā)控制。
列出常見的解決方案有:
使用隊列,額外起一個進程處理隊列,并發(fā)請求都放到隊列中,由額外進程串行處理,并發(fā)問題就不存在了,但是要額外進程支持以及處理延遲嚴重,本文不先不討論這種方法。
利用數(shù)據(jù)庫事務特征,做原子更新,此方法需要依賴數(shù)據(jù)庫的事務特性。
借助文件排他鎖,在處理下單請求的時候,用flock鎖定一個文件,成功拿到鎖的才能處理訂單。
一、利用 Redis 事務特征redis 事務是原子操作,可以保證訂單處理的過程中數(shù)據(jù)沒有被其它并發(fā)的進程修改。
示例代碼:
set(array( "reactor_num" => 2, //reactor thread num "worker_num" => 4 //worker process num )); $http->on("request", function (swoole_http_request $request, swoole_http_response $response) { $uniqid = uniqid("uid-", TRUE); // 模擬唯一用戶ID $redis = new Redis(); $redis->connect("127.0.0.1", 6379); // 連接 redis $redis->watch("rest_count"); // 監(jiān)測 rest_count 是否被其它的進程更改 $rest_count = intval($redis->get("rest_count")); // 模擬唯一訂單ID if($rest_count > 0){ $value = "{$rest_count}-{$uniqid}"; // 表示當前訂單,被當前用戶搶到了 // do something ... 主要是模擬用戶搶到單后可能要進行的一些密集運算 $rand = rand(100, 1000000); $sum=0; for ($i=0;$i<$rand;$i++){ $sum+=$i; } // redis 事務 $redis->multi(); $redis->lPush("uniqids", $value); $redis->decr("rest_count"); $replies = $redis->exec(); // 執(zhí)行以上 redis 事務 // 如果 rest_count 的值被其它的并發(fā)進程更改了,以上事務將回滾 if(!$replies){ echo "訂單 {$value} 回滾".PHP_EOL; } } $redis->unwatch(); }); $http->start();
使用 ab 測試
$ ab -t 20 -c 10 http://192.168.1.104:9509/二、利用文件排他鎖(阻塞模式)
阻塞模式下,如果進程在獲取文件排他鎖時,其它進程正在占用鎖的話,此進程會掛起等待其它進程釋放鎖后,并自己獲取到鎖后,再往下執(zhí)行。
示例代碼:
set(array( "reactor_num" => 2, //reactor thread num "worker_num" => 4 //worker process num )); $http->on("request", function (swoole_http_request $request, swoole_http_response $response) { $uniqid = uniqid("uid-", TRUE); $redis = new Redis(); $redis->connect("127.0.0.1", 6379); $fp = fopen("lock.txt", "w+"); // 阻塞(等待)模式, 要取得獨占鎖定(寫入的程序) if(flock($fp,LOCK_EX)) //鎖定當前指針 { // 成功取得鎖后,放心處理訂單 $rest_count = intval($redis->get("rest_count")); $value = "{$rest_count}-{$uniqid}"; if($rest_count > 0){ // do something ... $rand = rand(100, 1000000); $sum=0; for ($i=0;$i<$rand;$i++){ $sum+=$i; } $redis->lPush("uniqids", $value); $redis->decr("rest_count"); } // 訂單處理完成后,再釋放鎖 flock($fp,LOCK_UN); } fclose($fp); }); $http->start();
使用 ab 測試
$ ab -t 20 -c 10 http://192.168.1.104:9510/三、利用文件排他鎖(非阻塞模式)
非阻塞模式下,如果進程在獲取文件排他鎖時,其它進程正在占用鎖的話,此進程會馬上判斷獲取鎖失敗,并且繼續(xù)往下執(zhí)行。
示例代碼:
set(array( "reactor_num" => 2, //reactor thread num "worker_num" => 4 //worker process num )); $http->on("request", function (swoole_http_request $request, swoole_http_response $response) { $uniqid = uniqid("uid-", TRUE); $redis = new Redis(); $redis->connect("127.0.0.1", 6379); $fp = fopen("lock.txt", "w+"); // 非阻塞模式, 如果不希望 flock() 在鎖定時堵塞,則給 lock 加上 LOCK_NB if(flock($fp,LOCK_EX | LOCK_NB)) //鎖定當前指針 { // 成功取得鎖后,放心處理訂單 $rest_count = intval($redis->get("rest_count")); $value = "{$rest_count}-{$uniqid}"; if($rest_count > 0){ // do something ... $rand = rand(100, 1000000); $sum=0; for ($i=0;$i<$rand;$i++){ $sum+=$i; } $redis->lPush("uniqids", $value); $redis->decr("rest_count"); } // 訂單處理完成后,再釋放鎖 flock($fp,LOCK_UN); } else { // 如果獲取鎖失敗,馬上進入這里執(zhí)行 echo "{$uniqid} - 系統(tǒng)繁忙,請稍后再試".PHP_EOL; } fclose($fp); }); $http->start();
使用 ab 測試
$ ab -t 20 -c 10 http://192.168.1.104:9511/最后給出三種處理方式的測試結果比較
redis 事務方式:
...... Concurrency Level: 10 Time taken for tests: 20.005 seconds Complete requests: 17537 Failed requests: 0 Total transferred: 2578380 bytes HTML transferred: 0 bytes Requests per second: 876.62 [#/sec] (mean) Time per request: 11.407 [ms] (mean) Time per request: 1.141 [ms] (mean, across all concurrent requests) Transfer rate: 125.86 [Kbytes/sec] received ......
文件排他鎖(阻塞模式):
...... Concurrency Level: 10 Time taken for tests: 20.003 seconds Complete requests: 8205 Failed requests: 0 Total transferred: 1206282 bytes HTML transferred: 0 bytes Requests per second: 410.19 [#/sec] (mean) Time per request: 24.379 [ms] (mean) Time per request: 2.438 [ms] (mean, across all concurrent requests) Transfer rate: 58.89 [Kbytes/sec] received ......
文件排他鎖(非阻塞模式):
...... Concurrency Level: 10 Time taken for tests: 20.002 seconds Complete requests: 8616 Failed requests: 0 Total transferred: 1266846 bytes HTML transferred: 0 bytes Requests per second: 430.77 [#/sec] (mean) Time per request: 23.214 [ms] (mean) Time per request: 2.321 [ms] (mean, across all concurrent requests) Transfer rate: 61.85 [Kbytes/sec] received ......
經(jīng)測試結果對比,redis 事務方式優(yōu)于文件排他鎖方式,而文件排他鎖方式中,非阻塞模式優(yōu)于阻塞模式。
文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/31704.html
摘要:以下為大家整理了阿里巴巴史上最全的面試題,涉及大量面試知識點和相關試題。的內(nèi)存結構,和比例。多線程多線程的幾種實現(xiàn)方式,什么是線程安全。點擊這里有一套答案版的多線程試題。線上系統(tǒng)突然變得異常緩慢,你如何查找問題。 以下為大家整理了阿里巴巴史上最全的 Java 面試題,涉及大量 Java 面試知識點和相關試題。 JAVA基礎 JAVA中的幾種基本數(shù)據(jù)類型是什么,各自占用多少字節(jié)。 S...
摘要:對支持很好,分表后無需考慮全局的問題。但是這個項目使用的是進行開發(fā),必須自己生成全局。語句的是為了保證并發(fā)環(huán)境下的值只增不減。每次生成全局前,先檢測指定的是否存在。代碼如下另外對于全局的生成,和也都公布了自己的方案。 最近一個項目由于數(shù)據(jù)量變大,需要進行數(shù)據(jù)分表。數(shù)據(jù)存儲在淘寶的tddl上。分表后,原先的自增id就不能使用了。tddl對java支持很好,分表后無需考慮全局id的問題。但...
摘要:前言三年后端開發(fā)經(jīng)驗,面的目標崗位是的高級后端開發(fā)。面試結束,應該沒有后續(xù)。 前言 三年Java后端開發(fā)經(jīng)驗,面的目標崗位是20k-35k的高級后端Java開發(fā)。 第一場,基本裸面,關于曾經(jīng)的項目部門答的不好,所以還是得好好準備。 某C輪在線旅游公司 筆試 先做半個小時的筆試題,一共六個題目,兩道go語言的基礎題,一道斐波那契相關,一道數(shù)據(jù)庫行列轉置,一道實現(xiàn)一個棧,還有一道是百萬計...
摘要:話說當下一共有種運行模式,分別是和模塊模式。使用,全稱進程管理器進行管理。工作原理啟動時載入進程管理器進程管理器自身初始化,啟動多個解釋器進程并等待來自的連接當客戶端請求到達時,進程管理器選擇并連接到一個解釋器。 我們知道 workerman 程序需要在php-cli模式下運行,也就是命令行模式,這塊我們有必要了解一下。 話說PHP當下一共有4種運行模式,分別是CGI、FastCGI、...
摘要:為程序員金三銀四精心挑選的余道面試題與答案,歡迎大家向我推薦你在面試過程中遇到的問題我會把大家推薦的問題添加到下面的常用面試題清單中供大家參考。 為Java程序員金三銀四精心挑選的300余道Java面試題與答案,歡迎大家向我推薦你在面試過程中遇到的問題,我會把大家推薦的問題添加到下面的常用面試題清單中供大家參考。 前兩天寫的以下博客,大家比較認可,熱度不錯,希望可以幫到準備或者正在參加...
閱讀 3457·2021-11-17 17:00
閱讀 3818·2021-08-09 13:46
閱讀 2866·2019-08-30 15:54
閱讀 627·2019-08-30 13:54
閱讀 2945·2019-08-29 17:13
閱讀 3218·2019-08-29 14:00
閱讀 2975·2019-08-29 11:11
閱讀 1379·2019-08-26 10:15