一:首先,得簡單說說 thinkphp+workerman 的安裝。 安裝 thinkphp5.1 composer create-project topthink/think=5.1.x-dev tp5andworkman 安裝 think-worker 我的官方群點擊此處。 composer r ...
一:首先,得簡單說說 thinkphp+workerman 的安裝。
安裝 thinkphp5.1
composer create-project topthink/think=5.1.x-dev tp5andworkman
安裝 think-worker
我的官方群點擊此處。
composer require workerman/workerman
二:我們先看 think-worker 的代碼
- config/worker_server.php
- 先來個伺服器廣播消息的示例,每10秒鐘定時廣播一條消息
'onWorkerStart' => function ($worker) {
\Workerman\Lib\Timer::add(10, function()use($worker){
// 遍歷當前進程所有的客戶端連接,發送自定義消息
foreach($worker->connections as $connection){
$send['name'] = '系統信息';
$send['content'] = '這是一個定時任務信息';
$send['time'] = time();
$connection->send(json_encode($send));
}
});
}
但是在 onMessage 時,我們獲取不到 $worker 對象,所以無法廣播消息。
'onMessage' => function ($connection, $data) {
$origin = json_decode($data,true);
$send['name'] = '廣播數據';
$send['content'] = $origin['content'];
$message = json_encode($send);
foreach($worker->connections as $connection)
{
$connection->send($message);
}
}
嘗試了各種方法,貌似都不行
'onMessage' => function ($connection, $data)use($worker) {
// 這樣是獲取不到 $worker 對象的
// ...省略代碼
}
所以只能拋棄 thinkphp 給我們封裝的 think-worker 框架,得自己寫,(或者修改框架內部代碼)
修改框架內部的代碼:/vendor/topthink/think-worker/src/command/Server.php
,主要是把 onMessage 方法自己加進去
use() 就是把外部變數傳遞到函數內部使用,或者使用global $worker
$worker = new Worker($socket, $context);
$worker->onMessage = function ($connection, $data)use($worker) {
$origin = json_decode($data,true);
$send['name'] = '廣播數據';
$send['content'] = $origin['content'];
$send['uid'] = $connection->uid;
$message = json_encode($send);
foreach($worker->connections as $connection)
{
$connection->send($message);
}
};
這樣,我們就能夠獲取到 $worker 對象了
$worker->onMessage = function ($connection, $data)use($worker) { ... }
三:$connection 綁定 uid
其實你早都已經看出,$worker->connections 獲取到的是當前所有用戶的連接,connections 即為其中一個鏈接。
記錄websocket連接時間:
$worker->onConnect = function ($connection) {
$connection->login_time = time();
};
獲取websocket連接時間:
$worker->onMessage = function ($connection, $data)use($worker) {
$login_time = $connection->login_time;
};
由此可以看出,我們可以把數據綁定到 $connection 連接的一個屬性,例如:
$connection->uid = $uid;
當JavaScript端在連接websocket伺服器成功後,即把自己的 uid 立馬發送服務端綁定:
$worker->onMessage = function ($connection, $data)use($worker) {
$origin = json_decode($data,true);
if(array_key_exists('bind',$origin)){
$connection->uid = $origin['uid'];
}
};
四:單播發送消息,即自定義發送
$worker->onMessage = function ($connection, $data)use($worker) {
$origin = json_decode($data,true);
$sendTo = $origin['sendto']; // 需要發送的對方的uid
$content = $origin['content']; // 需要發送到對方的內容
foreach($worker->connections as $connection)
{
if( $connection->uid == $sendTo){
$connection->send($content);
}
}
};
到此,已經完成基於 workman 的自定義對象發送消息。
由於該php文件存放於composer中,只需要把該文件複製出來,放到application/command
,修改命名空間,即可保存到自己的項目中
五:對比swoole
1、workman可以在windows系統中運行,swoole則不能。
2、workman:$worker->connections獲取所有連接,$connection->id獲取自己的連接id;swoole:$server->connections獲取所有連接,$connection->fd獲取自己的連接id。
3、workman啟動時執行 onWorkerStart 方法,可以把定時器寫入到裡面;swoole 使用 WorkerStart 啟動定時器。
僅僅於聊天室或者定時器而言,workman 還是比較方便的。
以上內容希望幫助到大家,很多PHPer在進階的時候總會遇到一些問題和瓶頸,業務代碼寫多了沒有方向感,不知道該從那裡入手去提升,對此我整理了一些資料,包括但不限於:分散式架構、高可擴展、高性能、高併發、伺服器性能調優、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql優化、shell腳本、Docker、微服務、Nginx等多個知識點高級進階乾貨需要的可以免費分享給大家,需要的可以加入我的官方群點擊此處。