Swoole?webSocket消息服務(wù)系統(tǒng)代碼設(shè)計詳解
概述
已經(jīng)Swoole系列的第二篇知識點(diǎn)了,前一篇主要的針對處理的是方案設(shè)計,這一篇主要是代碼實(shí)現(xiàn)的內(nèi)容,主要介紹高性能的原因已經(jīng)實(shí)現(xiàn),編程框架使用EasySwoole。
Swoole 與 EasySwoole
Swoole屬于php中的一個超級擴(kuò)展,它會接管PHP的進(jìn)程,管理和分配worker,但他依賴 PHP-Cli模式。Swoole和Yaf有相似的地方,它們都是以守護(hù)進(jìn)程的模式、常駐內(nèi)存的方式達(dá)到提高處理性能。
Swoole內(nèi)置了TCP、UDP、WebSocket、協(xié)程、異步、Redis/Mysql鏈接池等高效開發(fā)手段和方法,當(dāng)然對于新模式也有新的挑戰(zhàn),比如swoole不能使用die,會致使worker滑落,協(xié)程使用后要注意使用回調(diào)的過程,如果對php的基礎(chǔ)知識不了解的同學(xué),可以翻看我之前的php博客里的總結(jié),這里就不過多敘述了。
EasySwoole是一款非常簡單上手易操作的Swoole框架,上線2年多以來經(jīng)得住生產(chǎn)環(huán)境的考驗(yàn),官網(wǎng)文檔寫的也很詳細(xì),還有專門解答的QQ群,大佬們會解答很多問題。
安裝EasySwoole框架
之前我使用composer進(jìn)行安裝的(composer是php中管理依賴包的工具,和node里面的npm,python的pip 一樣),現(xiàn)在都使用docker鏡像直接操作了。
docker pull easyswoole/easyswoole3 docker run -ti -p 9501:9501 -p 80:80 --name easyswoole easyswoole/easyswoole3
項(xiàng)目初始目錄結(jié)構(gòu):
├── EasySwooleEvent.php ├── bootstrap.php ├── composer.json ├── composer.lock ├── dev.php ├── easyswoole ├── produce.php └── vendor
注冊服務(wù) 與啟動加載
1.在EasySwooleEvent.php文件中加載初始化需要的Mysql、redis配置文件,所有需要的服務(wù)都需要在啟動文件中進(jìn)行注冊,才能使用。
public static function loadConf()
{
$ConfPath = EASYSWOOLE_ROOT . '/App/Conf/';
$Conf = Config::getInstance();
$files = File::scanDirectory($ConfPath);
if (!is_array($files['files'])) {
return;
}
foreach ($files['files'] as $file) {
$data = require_once $file;
$Conf->setConf(strtolower(basename($file, '.php')), (array)$data);
}
}
2.注冊Mysql連接池,Mysql連接池主要設(shè)置參數(shù),
<?php
use EasySwoole\ORM\Db\Config as DBConfig;
use EasySwoole\ORM\Db\Connection;
use EasySwoole\ORM\DbManager;
return [
//mysql數(shù)據(jù)庫配置
'mysql-msg' => [
'host' => '59.110.162.133',
'port' => '3306',
'database' => 'swoole_msg', //cpwxw2_db_v2
'username' => 'work',
'password' => 'cp2018csq123456',
'timeout' => 300,
'charset' => 'utf8mb4'
],
//Mysql連接池配置
'conn_pool' => [
'timeOut' => '3.0', //設(shè)置獲取連接池對象超時時間
'checkOut' => 30 * 1000, //設(shè)置檢測連接存活執(zhí)行回收和創(chuàng)建的周期
'maxidleTime' => 15, //連接池對象最大閑置時間(秒)
'maxObjectNumber' => 100, //設(shè)置最大連接池存在連接對象數(shù)量
'minObjectNumber' => 5, //設(shè)置最小連接池存在連接對象數(shù)量
'autoPing' => 5, //設(shè)置自動ping客戶端鏈接的間隔
],
];
3.注冊redis鏈接池
use \EasySwoole\Redis\Config\RedisConfig;
use \EasySwoole\RedisPool\RedisPool;
$redisConf = GlobalConfig::getInstance()->getConf('redis');
RedisPool::getInstance()->register(new RedisConfig($redisConf),'redis');
4.注冊自定義log
use \EasySwoole\EasySwoole\Logger; Logger::getInstance(new \App\Log\LogHandel());
5.我的項(xiàng)目里還使用了crontab模塊
use Swoole\Coroutine\Scheduler; use EasySwoole\EasySwoole\Crontab\Crontab; //用戶通知隊列 Crontab::getInstance()->addTask(\App\Crontab\PushUserNoticeMsg::class);
6.添加熱啟動
Swoole的服務(wù)屬于常駐內(nèi)存加載類型的服務(wù),所以每次修改代碼后都需要重啟服務(wù),所以為了方便,添加了熱加載目錄,熱加載原理就是當(dāng)檢測到指定目錄有代碼更新時,用傳遞信號的方式進(jìn)行,指揮進(jìn)程進(jìn)行重新加載。
$hotReloadOptions = new \EasySwoole\HotReload\HotReloadOptions; $hotReload = new \EasySwoole\HotReload\HotReload($hotReloadOptions); $hotReloadOptions->setMonitorFolder([EASYSWOOLE_ROOT . '/App']); $server = ServerManager::getInstance()->getSwooleServer(); $hotReload->attachToServer($server);
7.啟動/停止服務(wù),參數(shù)說明
-mode 說明啟動服務(wù)類型 -d 以守護(hù)進(jìn)程的方式
php easyswoole server start -mode=websocket -d php easyswoole server stop
異步和 DB(Redis/Mysql)使用
1.高性能的異步操作
如果是不依賴于結(jié)果的計算,異步操作提高性能的有效手段之一,異步操作不需要等待結(jié)果,更好的利用CPU和I/O傳輸。
use EasySwoole\EasySwoole\Task\TaskManager;
TaskManager::getInstance()->async(function () use ($tableName,$data) {
// todo code ...
});
- Redis/Mysql使用
在高并發(fā)情況下,資源浪費(fèi)的占用時間越短越好,可以提高程序的服務(wù)效率。在ORM默認(rèn)情況下是使用defer方法獲取pool內(nèi)的連接資源,并在協(xié)程退出時自動歸還,在此情況下,在帶來便利的同時,會造成不必要資源的浪費(fèi)。
我們可以使用invoke方式,讓ORM查詢結(jié)束后馬上歸還資源,可以提高資源的利用率。
Mysql的使用Demo.
DbManager::getInstance()->invoke(function (ClientInterface $client)
use ($where, $tableName, $count) {
$pushMsgModel = PushMsgModel::invoke($client);
$pushMsgObj = $pushMsgModel->tableName($tableName)
->field(['push_id'])
->get($where);
}, self::MYSQL_CONN_NAME);
Redis的使用Demo.
## invoke方式
public function setAuthorFd(int $uid, int $fd)
{
RedisPool::invoke(function (Redis $redis) use ($fd, $uid) {
$sRet = $redis->zAdd(self::PUSH_MSG_AUTHOR_NOTICE_SYSTEM, $uid, $fd);
}, self::REDIS_CONN_NAME);
}
## defer方式
$redis = \EasySwoole\RedisPool\RedisPool::defer('redis');
$data = $redis->lRange(self::PUSH_MSG_COMMENT_DELAY_LISTS, 0, 1000);
WebSocket服務(wù)
1.WebSocket協(xié)議是什么
WebSocket協(xié)議是基于TCP的一種新的網(wǎng)絡(luò)協(xié)議。它實(shí)現(xiàn)了瀏覽器與服務(wù)器全雙工(full-duplex)通信——允許服務(wù)器主動發(fā)送信息給客戶端。
在沒有WebSocket協(xié)議之前,在網(wǎng)頁中,實(shí)現(xiàn)一個聊天室只能使用ajax 不斷輪詢,請求服務(wù)器是否有數(shù)據(jù)產(chǎn)生,而這樣的實(shí)現(xiàn)方法會出現(xiàn)一系列的問題:
- 如果輪詢時間間隔太短,會導(dǎo)致客戶端和服務(wù)端在一個時間段內(nèi)不斷的進(jìn)行http tcp的握手/揮手動作和http 請求頭,響應(yīng)頭的傳輸,大量消耗服務(wù)器資源,如果用戶量大的情況,會造成服務(wù)器的繁忙以至于宕機(jī)
- 客戶端每次只能通過發(fā)送http 請求獲得服務(wù)器是否有數(shù)據(jù)返回,且數(shù)據(jù)的及時性無法保證
正因?yàn)樵谶@種情況下,所以WebSocket出現(xiàn)了,它只需要一次http握手,就可以保持一個長連接,使得服務(wù)器可以主動發(fā)送消息給客戶端,大大減少了輪詢機(jī)制的消耗。
2.WebSocket協(xié)議實(shí)現(xiàn)原理
在實(shí)現(xiàn)websocket連線過程中,需要通過瀏覽器發(fā)出websocket連線請求,然后服務(wù)器發(fā)出回應(yīng),這個過程通常稱為握手 。
在 WebSocket API,瀏覽器和服務(wù)器只需要做一個握手的動作,然后,瀏覽器和服務(wù)器之間就形成了一條快速通道。兩者之間就直接可以數(shù)據(jù)互相傳送。在此WebSocket 協(xié)議中,為我們實(shí)現(xiàn)即時服務(wù)帶來了兩大好處:
- Header: 互相溝通的Header是很小的-大概只有 2 Bytes
- Server Push: 服務(wù)器的推送,服務(wù)器不再被動的接收到瀏覽器的請求之后才返回數(shù)據(jù),而是在有新數(shù)據(jù)時就主動推送給瀏覽器。
//$fd,指的是系統(tǒng)里的文件描述符fd
use EasySwoole\EasySwoole\ServerManager;
$server = ServerManager::getInstance()->getSwooleServer();
$server->push($fd,json_encode($messageData));
//獲取當(dāng)前鏈接的詳細(xì)信息
$info = $server->getClientInfo($fd);
//獲取全部websocket中的鏈接fd
//全員在線消息通知
$server = ServerManager::getInstance()->getSwooleServer();
$start_fd = 0;
while(true)
{
$conn_list = $server->getClientList( $start_fd, $this->limit );
if ($conn_list===false || count($conn_list) === 0 || empty($conn_list))
{
break;
}
$start_fd = end($conn_list);
foreach ($conn_list as $fd){
$info = $server->getClientInfo($fd);
if ($info && $info['websocket_status'] === WEBSOCKET_STATUS_FRAME) {
$server->push($fd, json_encode($this->pushMsg));
}
}
}以上就是Swoole webSocket消息服務(wù)系統(tǒng)代碼設(shè)計詳解的詳細(xì)內(nèi)容,更多關(guān)于Swoole webSocket消息服務(wù)的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
laravel Model 執(zhí)行事務(wù)的實(shí)現(xiàn)
今天小編就為大家分享一篇laravel Model 執(zhí)行事務(wù)的實(shí)現(xiàn),具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-10-10
CI(CodeIgniter)框架中URL特殊字符處理與SQL注入隱患分析
這篇文章主要介紹了CI(CodeIgniter)框架中URL特殊字符處理與SQL注入隱患,結(jié)合實(shí)例形式分析了CodeIgniter框架中針對特殊字符的過濾及SQL注入隱患的相關(guān)原理,需要的朋友可以參考下2019-02-02
使用swoole 定時器變更超時未支付訂單狀態(tài)的解決方案
本文主要是借助 swoole 定時器和 redis 的 zset 來實(shí)現(xiàn)的定時檢查并過期未支付訂單,感興趣的朋友跟隨小編一起看看吧2019-07-07
destoon會員注冊提示“數(shù)據(jù)校驗(yàn)失?。?)”解決方法
這篇文章主要介紹了destoon會員注冊提示“數(shù)據(jù)校驗(yàn)失?。?)”解決方法,需要的朋友可以參考下2014-06-06
windows安裝composer并更換國內(nèi)鏡像的過程詳解
這篇文章主要給大家介紹了windows安裝composer并更換國內(nèi)鏡像的過程詳解,文中有詳細(xì)的流程介紹,通過圖文結(jié)合講解的非常詳細(xì),需要的朋友可以參考下2023-11-11

