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