亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

php解決注冊并發(fā)問題并提高QPS

瀏覽:3日期:2022-06-14 11:13:33
目錄前言一、環(huán)境要求二、下載框架和消息隊列中間件三、解決注冊重復提交四、消息隊列分解注冊功能五、通過命令運行消息隊列,以下以windows舉栗六、測試前言

前面在本地的windows通過apache的ab工具測試了600并發(fā)下“查詢指定手機是否存在再提交數據”的注冊功能會出現重復提交的情況,并且在注冊完成時還需要對邀請人進行獎勵,記錄邀請記錄,對該新用戶自動發(fā)布動態(tài)信息,發(fā)短信或發(fā)郵件等其他業(yè)務功能。所以這里當并發(fā)時,注冊功能就變得低效且容易出現問題。

先對重復提交的問題通過redis解決,再把注冊儲存用戶基本信息以后的操作放到隊列中進行異步執(zhí)行,可以很好的優(yōu)化注冊功能,提高QPS。

一、環(huán)境要求

PHP版本 >= 5.6.0

PHP框架:Thinkphp5.1.*

消息隊列:Think-queue2.0

PHP擴展:Redis

二、下載框架和消息隊列中間件下載tp5.1。composer create-project topthink/think=5.1.* tp5 --prefer-dist安裝think-queue。composer require topthink/think-queue php安裝redis擴展和打開redis服務端和客戶端。三、解決注冊重復提交

配置文件中cache設置為redis驅動,并新建控制器因為cache相關命名空間。

use think\Exception;use think\facade\Cache;use think\facade\Env;use think\Queue;使用無序集合存手機號,通過判斷當前手機號是否是在指定鍵里為成員(如果注冊存入數據庫失敗,通過sRem刪除該成員),然后再通過查詢數據庫判斷是否存在。private $cache;private $handler;// 實例化redispublic function __construct() { $this->cache = Cache::init(); $this->handler = $this->cache->handler();}// 判斷手機號是否在集合中$is_existe = $this->handler->sIsMember('register:mobile',$mobile);if(!$is_existe) { $this->handler->sAdd('register:mobile',$mobile);}else { //Log::write('---壓力測試'.date('Y-m-d h:i:s').'---手機號已存在'); var_dump('手機號已存在'); // 用戶已存在 die;}// 查詢手機號碼是否已注冊$user = db('user')->field('mobile')->where('mobile', $mobile)->find();if ($user) { //Log::write('---壓力測試'.date('Y-m-d h:i:s').'---手機號注冊了'); var_dump('手機號已注冊'); // 用戶已存在 die;}四、消息隊列分解注冊功能配置消息隊列,后面以redis驅動為例。<?phpreturn [ 'connector' => 'Redis', // Redis 驅動 'expire' => 60,// 任務的過期時間,默認為60秒; 若要禁用,則設置為 null 'default' => 'default',// 默認的隊列名稱 'host' => '127.0.0.1',// redis 主機ip 'port' => 6379, // redis 端口 'password' => '',// redis 密碼 'select' => 0,// 使用哪一個 db,默認為 db0 'timeout' => 0,// redis連接的超時時間 'persistent' => false, // 是否是長連接 // 'connector' => 'Database', // 數據庫驅動 // 'expire' => 60, // 任務的過期時間,默認為60秒; 若要禁用,則設置為 null // 'default' => 'default', // 默認的隊列名稱 // 'table' => 'jobs', // 存儲消息的表名,不帶前綴 // 'dsn' => [], // 'connector' => 'Topthink', // ThinkPHP內部的隊列通知服務平臺 ,本文不作介紹 // 'token' => '', // 'project_id' => '', // 'protocol' => 'https', // 'host'=> 'qns.topthink.com', // 'port'=> 443, // 'api_version' => 1, // 'max_retries' => 3, // 'default' => 'default',//'connector' => 'Sync',// Sync 驅動,該驅動的實際作用是取消消息隊列,還原為同步執(zhí)行];完成添加新用戶后將指定數據加入消息隊列。<?phpnamespace app\index\controller;use think\Db;use think\Validate;use think\Exception;use think\facade\Cache;use think\facade\Env;use think\Queue;use think\Log;class Index{private $cache; private $handler; public function __construct() {$this->cache = Cache::init();$this->handler = $this->cache->handler(); } public function index() {$data = input('post.');unset($data['balance']);unset($data['credit']);// $blacklist = [// '18124198164','13401363108','17688552009','15089352898','13602940094','13346643336','13181351655','18301123028','13598020751','13014568187',// '13428733909','17337991130','13275342497'// ];$rule = [ 'mobile' => 'require|number|length:11', 'password' => 'require|length:6,32',];$msg = [ 'mobile.require' => '手機號必須', 'mobile.length' => '手機號為11位數字', 'mobile.number' => '手機號為11位數字', 'password.require' => '密碼必須', 'password.length' => '密碼為6-12位之間',];//驗證數據是否合法$mobile = isset($data['mobile']) ? $data['mobile'] : '';$validate = new Validate($rule, $msg);$result = $validate->check($data);if (!$result) { var_dump($validate->getError()); die;}// if(in_array($mobile,$blacklist)) {// var_dump('該手機號已注冊了'); // 黑名單// die;// }// 判斷手機號是否在集合中$is_existe = $this->handler->sIsMember('register:mobile',$mobile);if(!$is_existe) { $this->handler->sAdd('register:mobile',$mobile);}else { //Log::write('---壓力測試'.date('Y-m-d h:i:s').'---手機號已存在'); var_dump('手機號已存在'); // 用戶已存在 die;}// 查詢手機號碼是否已注冊$user = db('user')->field('mobile')->where('mobile', $mobile)->find();if ($user) { //Log::write('---壓力測試'.date('Y-m-d h:i:s').'---手機號注冊了'); var_dump('手機號已注冊'); // 用戶已存在 die;} // 用戶不存在注冊// $data['id'] = getNewUserid();$data['no'] = date('Ymdhis').rand(100, 999);$data['avatar'] = 'https://rumcdn-1255484416.cos.ap-chengdu.myqcloud.com/img/d_h.png';$data['password'] = md5($data['password']);$randomNickname = date('Ymdhis').rand(100, 999); $data['nickname'] = 'rm_' . $randomNickname;$data['create_time'] = time();$data['type'] = 1;/***是否存在邀請人的跑步錢進號***/if(isset($data['pbqj_no']) && !empty($data['pbqj_no'])) { $inviter = db('user')->field('id')->where(['no'=>$data['pbqj_no']])->find(); if($inviter) {$data['inviter_id'] = $inviter['id']; }}/***是否存在邀請人的跑步錢進號***/unset($data['pbqj_no']);$userid = db('user')->insertGetId($data);if ($userid) {/******************加入消息隊列異步處理后續(xù)操作*******************/ // 1.當前任務將由哪個類來負責處理。 // 當輪到該任務時,系統(tǒng)將生成一個該類的實例,并調用其 fire 方法 $jobHandlerClassName = 'app\index\job\JobUser'; // 2.當前任務歸屬的隊列名稱,如果為新隊列,會自動創(chuàng)建 $jobQueueName = 'userJobQueue'; // 3.當前任務所需的業(yè)務數據 . 不能為 resource 類型,其他類型最終將轉化為json形式的字符串 // ( jobData 為對象時,需要在先在此處手動序列化,否則只存儲其public屬性的鍵值對) //$jobData = ['ts' => time(), 'bizId' => uniqid() , 'a' => 1]; $jobData = ['userid'=>$userid,'time'=>time(),'mobile'=>$mobile,'inviterid'=>(isset($data['inviter_id']) ? $data['inviter_id'] : 0)]; // 4.將該任務推送到消息隊列,等待對應的消費者去執(zhí)行 $isPushed = Queue::push($jobHandlerClassName , $jobData , $jobQueueName);// database 驅動時,返回值為 1|false ; redis 驅動時,返回值為 隨機字符串|false if($isPushed !== false) { var_dump('加入隊列成功');die;//Log::write('-----------加入消息隊列成功-----------');//echo date('Y-m-d H:i:s') . ' a new Hello Job is Pushed to the MQ'.''; }else{var_dump('加入消息隊列');die;//Log::write('-----------加入消息隊列失敗-----------');//echo 'Oops, something went wrong.'; }/******************加入消息隊列異步處理后續(xù)操作*******************/ $res['id'] = $userid; $res['no'] = $data['no']; // // token處理類 // $accessToken = new AccessToken(); // $accessToken = $accessToken->getToken($userid); // if (empty($accessToken)) { // //Log::write('---壓力測試'.date('Y-m-d h:i:s').'---秘鑰生成失敗'); // var_dump('秘鑰生成失敗'); // } else { // $res['user_token'] = $accessToken; // } // if (method_exists(\chat\User::class, 'getToken')) { // $chat_token = \chat\User::getToken($res['id'], $data['nickname'], $data['avatar']); // if (!$chat_token) { // //Log::write('---壓力測試'.date('Y-m-d h:i:s').'---聊天秘鑰生成失敗'); // var_dump('聊天秘鑰生成失敗'); // } else { // $res['chat_token'] = $chat_token; // } // } else { // $res['chat_token'] = ''; // } //Log::write('---壓力測試'.date('Y-m-d h:i:s').'---注冊成功'); var_dump($res); die;} else { //Log::write('---壓力測試'.date('Y-m-d h:i:s').'---數據庫錯誤'); $this->handler->sRem('register:mobile',$mobile); var_dump('數據庫錯誤'); die;} } public function hello($name = 'ThinkPHP5') {return 'hello,' . $name; }}

創(chuàng)建消費者(job),對執(zhí)行隊列中的任務。

(1). 在同一模塊下新建job文件夾和一個執(zhí)行類(JobUser), 需要對應生產者中jobHandlerClassName。

(2). 前面執(zhí)行完隊列加入成功后,可以本地使用redis客戶端通過lrange queues:userJobQueue 0 -1 查看隊列成員

(queues:userJobQueue中,userJobQueue是自己在加入隊列前自己起的隊列名稱,與queues: 拼接就是redis的list的鍵名,所以可以直接查看 )。

(3).隊列中的data就是自己傳遞的數據,后面需要在消費者中通過該數據進行注冊功能后的業(yè)務操作: 送獎勵,存儲邀請記錄,發(fā)動態(tài),發(fā)短信,發(fā)郵件等等。

<?phpnamespace app\index\job;use think\queue\Job;use think\Db;use think\Exception;use think\facade\Cache;use think\facade\Env;class JobUser { private $cache; private $handler; public function __construct() {$this->cache = Cache::init();$this->handler = $this->cache->handler(); } /** * fire方法是消息隊列默認調用的方法 * @param Job $job 當前的任務對象 * @param array|mixed $data 發(fā)布任務時自定義的數據 */ public function fire(Job $job,$data) {$job->delete();//print('hahah\n');// print('<info>The user already exists '.'</info>\n');// exit();if(empty($data) || empty($data['userid']) || empty($data['mobile'])) { $job->delete(); print('canshu buzu\n'); return;}// 如有必要,可以根據業(yè)務需求和數據庫中的最新數據,判斷該任務是否仍有必要執(zhí)行.$isJobStillNeedToBeDone = $this->checkDatabaseToSeeIfJobNeedToBeDone($data);if(!$isJobStillNeedToBeDone) { print('hahah\n'); $job->delete(); return;}$isJobDone = $this->doHelloJob($data);if ($isJobDone) { //如果任務執(zhí)行成功, 記得刪除任務 $job->delete(); print('<info>Hello Job has been done and deleted'.'</info>\n');}else{ if ($job->attempts() > 3) {//通過這個方法可以檢查這個任務已經重試了幾次了print('<warn>Hello Job has been retried more than 3 times!'.'</warn>\n');//$job->delete();// 也可以重新發(fā)布這個任務//print('<info>Hello Job will be availabe again after 2s.'.'</info>\n');//$job->release(2); //$delay為延遲時間,表示該任務延遲2秒后再執(zhí)行 }} } /** * 有些消息在到達消費者時,可能已經不再需要執(zhí)行了 * @param array|mixed $data 發(fā)布任務時自定義的數據 * @return boolean 任務執(zhí)行的結果 */ private function checkDatabaseToSeeIfJobNeedToBeDone($data) {// 判斷手機緩存集合中是否存在// $is_existe = $this->handler->sIsMember('register:mobile',$data['mobile']);// if($is_existe) {// return false; // } // // 查詢當前用戶是否在數據庫中存在// $userinfo = Db::name('user')->field('id')->where('id',$data['userid'])->find();// if($userinfo) {// return false; // } return true; } /** * 根據消息中的數據進行實際的業(yè)務處理 * @param array|mixed $data 發(fā)布任務時自定義的數據 * @return boolean 任務執(zhí)行的結果 */ private function doHelloJob($data) {try{ if(isset($data['inviterid']) && !empty($data['inviterid'])) {// 添加邀請記錄$res_record = Db::name('user_inviter') ->insert(['inviterid' => $data['inviterid'],'userid' => $data['userid'],'code'=> $data['inviterid'] . 'T' . $data['userid'],'create_time' => $data['time'],]);// 給邀請人贈送300步幣Db::name('user_credit') ->insert(['userid' => $data['inviterid'],'type'=> 1,'credit' => 300,'source' => $res_record,'create_time' => $data['time']]);// 更新邀請人步幣(用戶表)Db::name('user')->where('id', $data['inviterid'])->setInc('credit', 300); } { // 注冊成功發(fā)表動態(tài)$dynamic_data['userid'] = $data['userid'];$dynamic_data['dynamic'] = base64_encode('號外!號外!我加入跑步錢進了,大家一起走路領紅包吧!');$dynamic_data['images'][] = 'https://rumcdn-1255484416.cos.ap-chengdu.myqcloud.com/img/d_d.png';$dynamic_data['images'] = serialize($dynamic_data['images']);$dynamic_data['create_time'] = $data['time'];$result = Db::name('dynamic')->insert($dynamic_data); }}catch(\Exception $e) { Log::write('---執(zhí)行消息隊列出錯---'.$e->getMessage()); return false;}return true;// 根據消息中的數據進行實際的業(yè)務處理...//var_dump($data);//print('<info>Hello Job Started. job Data is: '.var_export($data,true).'</info> \n');//print('<info>Hello Job is Fired at ' . date('Y-m-d H:i:s') .'</info> \n');//print('<info>Hello Job is Done!'.'</info> \n');//return true; } /** * 該方法用于接收任務執(zhí)行失敗的通知,你可以發(fā)送郵件給相應的負責人員 * @param $jobData string|array|... //發(fā)布任務時傳遞的 jobData 數據 */ public function failed($jobData) {//send_mail_to_somebody() ;print('Warning: Job failed after max retries. job data is :'.var_export($jobData,true).'\n'); }}

(4). 設置任務執(zhí)行失敗后的處理,比如記錄日志或發(fā)郵件給開發(fā)者。

a. 在tags.php中配置失敗后執(zhí)行了類。

<?php// 應用行為擴展定義文件return [ // 應用初始化 'app_init' => [], // 應用開始 'app_begin' => [], // 模塊初始化 'module_init' => [], // 操作開始執(zhí)行 'action_begin' => [], // 視圖內容過濾 'view_filter' => [], // 日志寫入 'log_write' => [], // 應用結束 'app_end' => [], 'queue_failed' => [// 數組形式,[ 'ClassName' , 'methodName']['application\\behavior\\MyQueueFailedLogger', 'logAllFailedQueues']// 字符串(靜態(tài)方法),'StaicClassName::methodName'// 'MyQueueFailedLogger::logAllFailedQueues'// 字符串(對象方法),'ClassName',此時需在對應的ClassName類中添加一個名為 queueFailed 的方法// 'application\\behavior\\MyQueueFailedLogger'// 閉包形式/*function( &$jobObject , $extra){ // var_dump($jobObject); return true;}*/ ],];

b. 在application目錄下創(chuàng)建任務錯誤執(zhí)行后的處理腳本,根據業(yè)務需求自定。

<?phpnamespace app\behavior;use think\Db;class MyQueueFailedLogger{ const should_run_hook_callback = true; /** * @param $jobObject \think\queue\Job //任務對象,保存了該任務的執(zhí)行情況和業(yè)務數據 * @return bool true //是否需要刪除任務并觸發(fā)其failed() 方法 */ public function logAllFailedQueues(&$jobObject) {$failedJobLog = [ 'jobHandlerClassName' => $jobObject->getName(), // 'application\index\job\Hello' 'queueName' => $jobObject->getQueue(), // 'helloJobQueue' 'jobData' => $jobObject->getRawBody()['data'], // '{'a': 1 }' 'attempts' => $jobObject->attempts(), // 3];var_export(json_encode($failedJobLog,true));$data = [ 'content' => json_encode($failedJobLog,true), 'create_time' => time(),];Db::name('ztest')->insertGetId($data);// $jobObject->release(); //重發(fā)任務//$jobObject->delete(); //刪除任務//$jobObject->failed(); //通知消費者類任務執(zhí)行失敗return self::should_run_hook_callback; }}五、通過命令運行消息隊列,以下以windows舉栗cmd進入當前項目, 然后輸入 "php think queue:listen --queue userJobQueue" (userJobQueue是自己的隊列名)。也可以在項目的根目錄創(chuàng)建bat文件,文件寫入"php think queue:listen --queue userJobQueue",保存只需雙擊就可以執(zhí)行。 六、測試

結果使用了消息隊列后,同樣610的并發(fā),使用時間就縮短了

以上就是php解決注冊并發(fā)問題并提高QPS的詳細內容,更多關于php注冊并發(fā)提高QPS的資料請關注好吧啦網其它相關文章!

標簽: PHP
主站蜘蛛池模板: 一级黄色片一级黄色片 | 一级毛片成人午夜 | 狠狠色丁香婷婷综合最新地址 | 久cao在线香蕉69影院 | 美女视频大全视频a免费九 美女视频大全美女视频黄 美女视频毛片 | 国产伦精品一区二区三区无广告 | 亚洲国产精品一区二区三区久久 | 91高清在线成人免费观看 | 国产福利在线观看精品 | 高清成年美女xx免费网站黄 | 国产人成77777视频网站 | 亚洲精品国产一区二区三区在 | 国产精品果冻麻豆精东天美 | 国产亚洲一区二区精品 | 国产亚洲福利精品一区二区 | 爱爱小视频在线观看网站 | 午夜一区二区三区 | 一级α一级α片免费观看网站 | 秘书上班和老板啪啪中文字幕 | 成人精品视频一区二区在线 | 免费黄在线看 | 一级欧美毛片成人 | 欧美日韩黄色 | 在线亚洲欧美日韩 | 91久久亚洲精品国产一区二区 | 亚洲福利精品 | 免费看片aⅴ免费大片 | 国外欧美一区另类中文字幕 | 狠狠色丁婷婷综合久久 | 久久福利青草狠狠午夜 | 国产成人精选免费视频 | 欧美一级毛片欧美一级 | 欧美国产一区二区三区 | 国产在线精品福利91香蕉 | 一级特一级特色生活片 | 视频一区国产 | 国产综合精品久久久久成人影 | 国产视频在线免费观看 | 久久99热不卡精品免费观看 | 久久国产精品亚洲一区二区 | 亚洲大成色www永久网址 |