WebSocket 入坑記

為了在小程序中實現(xiàn)即時消息的提示,決定入坑WebSocket

  1. 小程序uni-app
  2. 后端Thnikphp

#01. 安裝Workermantopthink/think-worker

Workerman是一個純PHP的實現(xiàn),因此基本上不需要特殊的安裝,你只需要通過composer直接安裝即可。

ThinkPHP 5.1 Workerman 快速上手指南 · ThinkPHP5.1 Workerman上手指南 · 看云 (kancloud.cn)

// 安裝`Workerman`
composer require workerman/workerman

// 安裝`think-worker`擴展,如果你還沒有安裝`Workerman` 的話,安裝擴展同時會自動安裝
composer require topthink/think-worker
// 安裝完成后會在tp 的`config`目錄下生成`gateway_worker.php`,`worker.php`,`worker_server.php` 三個配置文件

#02. 啟動Workerman 服務

  1. Workerman HTTP 服務(略)
php think worker
  1. Workerman Server 服務(本文主要使用WebSocket)

關(guān)于WebSocket 詳細的說明:python - WebSocket 通信過程與實現(xiàn) - Geek情懷 - SegmentFault 思否

php think worker:server

#03. worker_server.php 配置

worker_server.php 配置說明:Workerman · ThinkPHP6.0完全開發(fā)手冊 · 看云 (kancloud.cn)

#04. 自定義服務類

自定義服務類必須繼承think\worker\Server類,支持workerman所有的回調(diào)方法定義(回調(diào)方法必須是public類型)。

#04.01 worker_server.php 配置參數(shù)
return [
    'worker_class'  =>  'app\http\Worker',
];
#04.02 自定義服務類
<?php
namespace app\http;

use think\facade\Env;
use think\worker\Server;

class Worker extends Server
{
    protected $host = '127.0.0.1';
    protected $port = 2346;
    protected $option = [ 
        'count'     => 4,
        'pidFile'   => Env::get('runtime_path') . 'worker.pid', // 坑#01
        'name'      => 'think'
    ];

    public function onMessage($connection, $data)
    {
        $connection->send('receive success');
    }
}

坑#01 Accessing static property Workerman\Worker::$pidFile as non static

開端:$pidFile,文件指定保存位置
原因:think-worker 擴展中Server 類,關(guān)于Workerman\Woker 類的靜態(tài)屬性$pidFile 作為對象屬性 來處理

// 部分代碼
    public function __construct()
    {
        // 實例化 Websocket 服務
        $this->worker = new Worker($this->socket ?: $this->protocol . '://' . $this->host . ':' . $this->port, $this->context);

        // 設置參數(shù)($opthin 的處理)
        if (!empty($this->option)) {
            foreach ($this->option as $key => $val) {
                $this->worker->$key = $val; // **坑#01**
            }
        }
    }

解決

// 自定義服務類代碼
    public function init()
    {
        parent::init();
        \Workerman\Worker::$pidFile = app()->getRuntimePath() . 'workerman.pid'; // `靜態(tài)屬性` 賦值
    }

坑#02 無法通過自定義頭信息header 來傳遞數(shù)據(jù),例如:token

坑之說明:前端websocket請求頭問題 · Issue #7 · DvcLAB/DvcAI-fe · GitHub

只能通過手動ws.send(token) 來帶上token

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容