摘要:以一個較低的成本將傳統(tǒng)業(yè)務(wù)邏輯轉(zhuǎn)換成非阻塞高可用可擴展的異步模式。給付費的大爺走隊列非阻塞式重試最多重試次恭喜你注冊成功請點擊郵件中的鏈接完成驗證應(yīng)用和解惑我們采用的是開源的來為我們提供的服務(wù)。
async-helper 簡介
PHP 的異步進程助手,借助于 AMQP 實現(xiàn)異步執(zhí)行 PHP 的方法,將一些很耗時、追求高可用、需要重試機制的操作放到異步進程中去執(zhí)行,將你的 HTTP 服務(wù)從繁重的業(yè)務(wù)邏輯中解脫出來。以一個較低的成本將傳統(tǒng) PHP 業(yè)務(wù)邏輯轉(zhuǎn)換成非阻塞、高可用、可擴展的異步模式。
依賴php 5.6+
ext-bcmath
ext-amqp 1.9.1+
ext-memcached 3.0.3+
安裝通過 composer 安裝
composer require l669/async-helper
或直接下載項目源碼
wget https://github.com/l669306630/async-helper/archive/master.zip使用范例
業(yè)務(wù)邏輯:這里定義了很多等待被調(diào)用的類和方法,在你的項目中這可能是數(shù)據(jù)模型、或是一個發(fā)送郵件的類。
生產(chǎn)者:通常是 HTTP 服務(wù),傳統(tǒng)的 PHP 項目或是一個命令行程序,接收到某個請求或指令后進行一系列的操作。
"127.0.0.1", "port" => "5672", "user" => "root", "pass" => "123456", "vhost" => "/" ]); // 第三步、保存用戶信息到數(shù)據(jù)庫 $mail = [ "from" => "[email protected]", "to" => "[email protected]", "subject" => "恭喜你注冊成功", "body" => "請點擊郵件中的鏈接完成驗證...." ]; // 第四步、通過異步助手發(fā)送郵件 $async_helper->run("SendMailHelper", "request", [$mail]); // 這是同步的模式去發(fā)送郵件,如果郵件服務(wù)響應(yīng)遲緩或異常,就會直接影響該請求的響應(yīng)時間,甚至丟失這封重要郵件 // SendMailHelper::request($mail); } }消費者:PHP 的異步進程,監(jiān)聽消息隊列,執(zhí)行你指定的方法。并且該消費者進程是可擴展的高可用的服務(wù),這一切都得益于 AMQP,這是系統(tǒng)解耦、布局微服務(wù)的最佳方案。
consume.php
"127.0.0.1", "port" => "5672", "user" => "root", "pass" => "123456", "vhost" => "/", "cacheHelper" => $cache_helper ]); $async_helper->consume(); }catch(Exception $e){ // 可以在這里記錄一些日志 sleep(2); } }# 在命令行下啟動消費者進程,推薦使用 supervisor 來管理進程 php consume.php支持事務(wù):需要一次提交執(zhí)行多個異步方法,事務(wù)可以確保完成性。
// 接著上面的示例來說,這里省略了一些重復(fù)的代碼,下同 $async_helper->beginTransaction(); try{ $async_helper->run("SendMailHelper", "request", [$mail1]); $async_helper->run("SendMailHelper", "request", [$mail2]); $async_helper->run("SendMailHelper", "request", [$mail3]); $async_helper->commit(); }catch(Exception $e){ $async_helper->rollback(); }阻塞式重試:當(dāng)異步進程執(zhí)行一個方法,方法內(nèi)部拋出異常時進行重試,一些必須遵循執(zhí)行順序的業(yè)務(wù)就要采用阻塞式的重試,通過指定重試最大阻塞時長來控制。
use l669CacheHelper; use l669AsyncHelper; $async_helper = new AsyncHelper([ "host" => "127.0.0.1", "port" => "5672", "user" => "root", "pass" => "123456", "vhost" => "/", "cacheHelper" => new CacheHelper("127.0.0.1", 11211), "retryMode" => AsyncHelper::RETRY_MODE_REJECT, // 阻塞式重試 "maxDuration" => 600 // 最長重試 10 分鐘 ]); $send_mail_helper = new SendMailHelper(); $mail = new stdClass(); $mail->from = "[email protected]"; $mail->to = "[email protected]"; $mail->subject = "恭喜你注冊成功"; $mail->body = "請點擊郵件中的鏈接完成驗證...."; $async_helper->run($send_mail_helper, "request", [$mail]); // 如果方法中需要拋出異常來結(jié)束程序,又不希望被異步進程重試,可以拋出以下幾種錯誤碼,進程捕獲到這些異常后會放棄重試: // l669AsyncException::PARAMS_ERROR // l669AsyncException::METHOD_DOES_NOT_EXIST // l669AsyncException::KNOWN_ERROR非阻塞式重試:當(dāng)異步執(zhí)行的方法內(nèi)部拋出異常,async-helper 會將該方法重新放進隊列的尾部,先執(zhí)行新進入隊列的方法,回頭再重試剛才執(zhí)行失敗的方法,通過指定最大重試次數(shù)來控制。
use l669CacheHelper; use l669AsyncHelper; $async_helper = new AsyncHelper([ "host" => "127.0.0.1", "port" => "5672", "user" => "root", "pass" => "123456", "vhost" => "new", "cacheHelper" => new CacheHelper("127.0.0.1", 11211), "queueName" => "emails.vip", // 給付費的大爺走 VIP 隊列 "retryMode" => AsyncHelper::RETRY_MODE_TTL, // 非阻塞式重試 "maxRetries" => 10 // 最多重試 10 次 ]); $mail = new stdClass(); $mail->from = "[email protected]"; $mail->to = "[email protected]"; $mail->subject = "恭喜你注冊成功"; $mail->body = "請點擊郵件中的鏈接完成驗證...."; $async_helper->run("SendMailHelper", "request", [$mail]);應(yīng)用和解惑我們采用的是開源的 RabbitMQ 來為我們提供的 AMQP 服務(wù)。
你的項目部署在擁有很多服務(wù)器節(jié)點的集群上,每個節(jié)點的程序都需要寫日志文件,現(xiàn)在的問題就是要收集所有節(jié)點上面的日志到一個地方,方便我們及時發(fā)現(xiàn)問題或是做一些統(tǒng)計。所有節(jié)點都可以使用 async-helper 異步調(diào)用一個寫日志的方法,而執(zhí)行這個寫日志的方法的進程只需要在一臺機器上啟動就可以了,這樣所有節(jié)點的日志就都實時掌握在手里了。
做過微信公眾號開發(fā)的都知道,騰訊微信可以將用戶的消息推送到我們的服務(wù)器,如果我們在 5s 內(nèi)未及時響應(yīng),騰訊微信會重試 3 次,其實這就是消息隊列的應(yīng)用,使用 async-helper 可以輕松的做和這一樣的事情。
得益于 RabbitMQ,你可以輕松的橫向擴展你的消費者進程的能力,因為 RabbitMQ 天生就支持集群部署,你可以輕松的啟動多個消費者進程,或是將消費者進程分布到多臺機器上。
如果 RabbitMQ 服務(wù)不可用怎么辦呢?部署 RabbitMQ 高可用服務(wù)是容易的,對外提供單一 IP,這個 IP 是個負(fù)載均衡,背后是 RabbitMQ 集群,負(fù)載均衡承擔(dān)對后端集群節(jié)點的健康檢查。
async-helper 能否承受高并發(fā)請求?async-helper 生產(chǎn)者使用的是短連接,也就說在你的 HTTP 還沒有響應(yīng)瀏覽器的時候 async-helper 就已經(jīng)結(jié)束了工作,你連接 RabbitMQ 的時間是百分之百小于 HTTP 請求的時間的,換言之,只要 RabbitMQ 承受并發(fā)的能力超過你的 HTTP 服務(wù)的承受并發(fā)的能力,RabbitMQ 就永遠(yuǎn)不會崩,通過橫向擴展 RabbitMQ 很容易做到的。
和傳統(tǒng) PHP 相比對任何 PHP 方法通過反射進行異步執(zhí)行;
高可用,執(zhí)行方法進入消息隊列,可持久化,即使服務(wù)器宕機,執(zhí)行任務(wù)也不丟失;
高可用,對異??梢赃M行不限次數(shù)和時間的重試,重試次數(shù)和時間可配置;
支持對多個異步方法包含在事務(wù)中執(zhí)行,支持回滾事務(wù);
方法的參數(shù)類型支持除資源類型(resource)和回調(diào)函數(shù)(callable)外的任意類型的參數(shù);
得益于 AMQP,異步方法可以承受高并發(fā)、高負(fù)載,支持集群部署、橫向擴展;
低延時,實測延時時間 0.016 ~ 0.021s;
適用于:日常數(shù)據(jù)庫操作、日志收集、金融交易、消息推送、發(fā)送郵件和短信、數(shù)據(jù)導(dǎo)入導(dǎo)出、計算大量數(shù)據(jù)生成報表;
附錄安裝 memcached
安裝 rabbitmq
安裝 php7.1.5、ext-amqp、ext-memcached
安裝 supervisor
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/28219.html
摘要:下文如無特殊聲明將使用進程同時表示進程線程。收到數(shù)據(jù)后服務(wù)器程序進行處理然后使用向客戶端發(fā)送響應(yīng)。現(xiàn)在各種高并發(fā)異步的服務(wù)器程序都是基于實現(xiàn)的,比如。 并發(fā) IO 問題一直是服務(wù)器端編程中的技術(shù)難題,從最早的同步阻塞直接 Fork 進程,到 Worker 進程池/線程池,到現(xiàn)在的異步IO、協(xié)程。PHP 程序員因為有強大的 LAMP 框架,對這類底層方面的知識知之甚少,本文目的就是詳細(xì)介...
摘要:定義任務(wù)處理方法。讀取來自命令行的參數(shù),開始執(zhí)行任務(wù)。該函數(shù)有兩個參數(shù)和,是引用類型,用來存儲子進程的狀態(tài),有兩個可選常量,分別表示不等待子進程結(jié)束立即返回和等待子進程結(jié)束。 用PHP來實現(xiàn)異步任務(wù)一直是個難題,現(xiàn)有的解決方案中:PHP知名的異步框架有 swoole 和 Workerman,但都是無法在 web 環(huán)境中直接使用的,即便強行搭建 web 環(huán)境,異步調(diào)用也是使用多進程模式實...
摘要:此問題稱為查詢問題。您將只執(zhí)行兩個查詢而不是這是巨大的性能提升。這項工作是通過從數(shù)據(jù)庫中執(zhí)行查詢完成的查詢可能涉及到表以及其他的一些表。比如查詢,視圖,時間等等另一個非常酷的工具是,對應(yīng)用,有優(yōu)雅的調(diào)試助手的美稱。 showImg(https://segmentfault.com/img/remote/1460000018339917?w=1280&h=722); 讓我們開始吧!假若你...
摘要:實現(xiàn)異步的方式有哪些提供了一些異步方法那它們底層是用哪種方式實現(xiàn)的呢實現(xiàn)異步的方式有線程方式進程方式復(fù)用線程方式和進程方式類似有異步請求時開一個線程或者進程獲取到數(shù)據(jù)后線程間可以直接共享數(shù)據(jù)進程間可以通過進程通信機制,如共享內(nèi)存管道等方式進 實現(xiàn)異步的方式有哪些? swoole提供了一些異步方法, 那它們底層是用哪種方式實現(xiàn)的呢? 實現(xiàn)異步的方式有: 線程方式 進程方式 IO復(fù)用 ...
摘要:如果僅依靠程序自動交出控制的話,那么一些惡意程序?qū)苋菀渍加萌繒r間而不與其他任務(wù)共享。多個操作可以在重疊的時間段內(nèi)進行。 PHP下的異步嘗試系列 如果你還不太了解PHP下的生成器,你可以根據(jù)下面目錄翻閱 PHP下的異步嘗試一:初識生成器 PHP下的異步嘗試二:初識協(xié)程 PHP下的異步嘗試三:協(xié)程的PHP版thunkify自動執(zhí)行器 PHP下的異步嘗試四:PHP版的Promise ...
閱讀 2322·2021-11-24 09:39
閱讀 3055·2021-10-15 09:39
閱讀 3106·2021-07-26 23:38
閱讀 2301·2019-08-30 11:14
閱讀 3420·2019-08-29 16:39
閱讀 1723·2019-08-29 15:23
閱讀 791·2019-08-29 13:01
閱讀 2673·2019-08-29 12:29