溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Think-Swoole之WebSocket-Room加入、離開房間和房間消息發(fā)送的案例

發(fā)布時間:2021-03-09 15:17:25 來源:億速云 閱讀:201 作者:小新 欄目:編程語言

這篇文章主要介紹了Think-Swoole之WebSocket-Room加入、離開房間和房間消息發(fā)送的案例,具有一定借鑒價值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。

Think-Swoole 3.0 中 Websocket 新增了 Room 聊天室功能,它主要用于群發(fā)消息,但不同Room之間的消息又是相互隔離的。當我們進入一個聊天室,那么我們的進入、離開以及發(fā)送的消息只有這個聊天室的 fd 能接收到。

config.swoole.php

'websocket'  => [
        'enable'        => true,
        'handler'       => Handler::class,
        'parser'        => Parser::class,
        'ping_interval' => 25000,
        'ping_timeout'  => 60000,
        'room'          => [
            'type'  => 'table',
            'table' => [
                'room_rows'   => 4096,
                'room_size'   => 2048,
                'client_rows' => 8192,
                'client_size' => 2048,
            ],
            'redis' => [
                'host'          => '127.0.0.1',
                'port'          => 6379,
                'max_active'    => 3,
                'max_wait_time' => 5,
            ],
        ],
        'listen'        => [],
        'subscribe'     => [],
    ],

其中有 room 配置項,里面的 type 表示使用哪種數據處理方式,下面有兩種,“table” 和“redis”,table 是可以直接拿來使用的,而 redis 則需要我們的系統(tǒng)和項目中安裝了 redis 擴展。table 是一種高性能、跨進程的內存處理服務,不同進程間可以共享數據。

創(chuàng)建事件

在項目根目錄輸入如下命令,分別創(chuàng)建加入房間事件、離開房間事件和房間的聊天事件:

php think make:listener WsJoin
php think make:listener WsLeave
php think make:listener RoomTest

然后在 app/event.php 中定義事件:

[
    ],
    'listen'    => [
        'AppInit'  => [],
        'HttpRun'  => [],
        'HttpEnd'  => [],
        'LogLevel' => [],
        'LogWrite' => [],
        //監(jiān)聽連接,swoole 事件必須以 swoole 開頭
        'swoole.websocket.Connect' => [
            app\listener\WsConnect::class
        ],
        //監(jiān)聽關閉
        'swoole.websocket.Close' => [
            \app\listener\WsClose::class
        ],
        //監(jiān)聽 Test 場景
        'swoole.websocket.Test' => [
            \app\listener\WsTest::class
        ],
        //加入房間事件
        'swoole.websocket.Join' => [
            \app\listener\WsJoin::class
        ],
        //離開房間事件
        'swoole.websocket.Leave' => [
            \app\listener\WsLeave::class
        ],
        //處理聊天室消息
        'swoole.websocket.RoomTest' => [
            \app\listener\RoomTest::class
        ],
    ],
    'subscribe' => [
    ],
];

上述的 Join、Leave和RoomTest等名稱都是自定義的,要和前端發(fā)送消息的場景值對應。

當然,事件定義一樣可以在 config/swoole.php 配置文件的 websocket listen 進行配置,具體參考前面文章。

H5 WebSocker 客戶端方式連接

wsroot.html

<!DOCTYPE HTML>
<html>
<head>
    <meta charset="UTF-8">
    <title>Document</title>
</head>
<body>
<button onclick="join()">加入房間</button>
<button onclick="leave()">離開房間</button>
<input type="text" id="message">
<button onclick="send()">發(fā)送</button>
<script>
    var ws = new WebSocket("ws://127.0.0.1:9501/?uid=1");
    ws.onopen = function(){
        console.log('連接成功');
    }
    //數據返回的解析
    function mycallback(data){
        var start = data.indexOf('[') // 第一次出現(xiàn)的位置
        var start1 = data.indexOf('{')
        if(start < 0){
            start = start1;
        }
        if(start >= 0 && start1 >= 0){
            start = Math.min(start,start1);
        }
        if(start >= 0){
            console.log(data);
            var json = data.substr(start); //截取
            var json = JSON.parse(json);
            console.log(json);
            // if(json instanceof Array){
            //     window[json[0]](json[1]);
            // }
        }
    }
    function sendfd($message){
        console.log($message)
    }
    function testcallback($message){
        console.log($message)
    }
    function joincallback($message){
        // console.log($message)
        console.log(11);
    }
    function leavecallback($message){
        console.log($message)
    }
    ws.onmessage = function(data){
        // console.log(data.data);
        mycallback(data.data);
    }
    ws.onclose = function(){
        console.log('連接斷開');
    }
    function join()
{
        var room = prompt('請輸入房間號');
        ws.send(JSON.stringify(['join',{
            room:room
        }])); //發(fā)送的數據必須是 ['test',數據] 這種格式
    }
    function leave()
{
        var room = prompt('請輸入要離開的房間號');
        ws.send(JSON.stringify(['leave',{
            room:room
        }])); //發(fā)送的數據必須是 ['test',數據] 這種格式
    }
    function send()
{
        var message = document.getElementById('message').value;
        var room = prompt('請輸入接收消息的房間號')
        ws.send(JSON.stringify(['RoomTest',{
            message:message,
            room:room
        }])); //發(fā)送的數據必須是 ['test',數據] 這種格式
    }
</script>
</body>
</html>

SocketIO 客戶端方式連接

ioroomtest.html

<!DOCTYPE HTML>
<html>
<head>
    <meta charset="UTF-8">
    <title>Document</title>
</head>
<body>
<button onclick="join()">加入房間</button>
<button onclick="leave()">離開房間</button>
<input type="text" id="message">
<button onclick="send()">發(fā)送</button>
<script src="./socketio.js"></script>
<script>
    //http 協(xié)議
    var socket = io("http://127.0.0.1:9501?uid=1", {transports: ['websocket']});
    socket.on('connect', function(){
        console.log('connect success');
    });
    socket.on('close',function(){
       console.log('connect close')
    });
    //send_fd 為自定義的場景值,和后端對應
    socket.on("sendfd", function (data) {
        console.log(data)
    });
    //testcallback 為自定義的場景值,和后端對應
    socket.on("testcallback", function (data) {
        console.log(data)
    });
    socket.on("joincallback", function (data) {
        console.log(data)
    });
    socket.on("roomtestcallback", function (data) {
        console.log(data)
    });
    function join()
{
        var room = prompt('請輸入房間號');
        socket.emit('join',{
            room : room
        });
    }
    function leave()
{
        var room = prompt('請輸入要離開的房間號');
        socket.emit('leave',{
            room : room
        });
    }
    function send()
{
        var message = document.getElementById('message').value;
        var room = prompt('請輸入接收消息的房間號')
        socket.emit('RoomTest',{
            message : message,
            room : room
        });
    }
</script>
</body>
</html>

頁面中,join()、leave()、和send() 函數中定義的場景值分別是 join、leave和RoomTest,我們在 app/leave.php 中定義了這些場景值對應的事件,因此分別觸發(fā) WsJoin.php、WsLeave.php 和 RoomTest.php 事件。

后端事件編寫

app/listener/WsJoin.php

<?php
declare (strict_types = 1);
namespace app\listener;
class WsJoin
{
    /**
     * 事件監(jiān)聽處理
     *
     * @return mixed
     */
    public function handle($event)
{
        $ws = app('think\swoole\Websocket');
        $roomobj = app('think\swoole\websocket\Room');
        //當前客戶端加入指定 Room
        $ws -> join($event['room']);
        //同時加入多個房間
//        $ws -> join(['room1','room2']);
        //指定客戶端加入指定 room
//        $ws -> setSender(2) -> join($event['room']);
        //獲取當前房間所有的 fd
        $getAllFdInRoom = $roomobj -> getClients($event['room']);
        //獲取指定 fd 加入哪些房間
        $getAllRoom = $roomobj -> getRooms($ws -> getSender());
        var_dump('當前房間所有 fd:',$getAllFdInRoom);
        var_dump('當前 fd 加入的所有房間:',$getAllRoom);
        var_dump('當前請求數據:',$event);
        $ws -> emit('joincallback','房間加入成功');
    }
}

app/listener/WsLeave.php

<?php
declare (strict_types = 1);
namespace app\listener;
class WsLeave
{
    /**
     * 事件監(jiān)聽處理
     *
     * @return mixed
     */
    public function handle($event)
{
        $ws = app('think\swoole\Websocket');
        $roomobj = app('think\swoole\websocket\Room');
        // 當前客戶端離開指定 room
        $ws -> leave($event['room']);
        // 同時離開多個 room
//        $ws -> leave(['one','two']);
        // 指定客戶端離開指定 room
//        $ws -> setSender(2) -> leave($event['room']);
        // 獲取指定 room 中的所有客戶端 fd
        $getAllFdInRoom = $roomobj -> getClients($event['room']);
        var_dump('當前房間還剩 fd:',$getAllFdInRoom);
        $ws -> emit('leavecallback','房間離開成功');
    }
}

app/listener/RoomTest.php

<?php
declare (strict_types = 1);
namespace app\listener;
class RoomTest
{
    /**
     * 事件監(jiān)聽處理
     *
     * @return mixed
     */
    public function handle($event)
{
        var_dump($event);
        $ws = app('think\swoole\Websocket');
        //給指定的 room 內所有 fd 發(fā)送消息,包括當前客戶端,當前客戶端沒有加入該 room 也可發(fā)送
        $ws -> to($event['room']) -> emit('roomtestcallback',$event['message']);
        //指定多個 room 發(fā)送消息
        //$ws -> to(['one','two']) -> emit('客戶端場景值',$event['message']);
    }
}

以上分別是前端 HTML 頁面和后端的加入房間、離開房間和房間聊天事件代碼,下面開始測試。

首先在項目根目錄開啟 Think-Swoole 服務。

瀏覽器訪問 wsroot.html 或者 ioroomtest.html 頁面,可以打開多個標簽,模擬多個客戶端,我們先打開三個,連接成功后,fd 分別為1、2、3,我們讓 fd 1 和 2 的客戶端都加入“one”,房間,fd 為 3 的客戶端加入 “two” 房間,由于我們在 WsJoin.php 加入房間事件中打印了用戶加入房間所有 fd,和該用戶所加入的所有房間名稱,這些信息會打印在命令行中,最后會發(fā)送給當前客戶端聊天場景值和“加入房間成功”信息給客戶端,在瀏覽器控制臺可查看。

加入房間后,用 fd 為 1 的客戶端在頁面的輸入框輸入要發(fā)送的消息,點擊發(fā)送后,輸入要發(fā)送的房間名稱為“one”,然后,消息就發(fā)送到“one”房間,只有“one”房間的所有客戶端(fd 為1、2)能收到消息。

現(xiàn)在讓 fd 為 2 的客戶端離開 “one” 房間,由于在 WsLeave.php 離開房間事件中,我們打印了離開后剩余 fd,信息會出現(xiàn)在命令行中,可見“one”房間只剩下 fd 1 了,fd 1 發(fā)送信息,fd 2 就收不到了。

感謝你能夠認真閱讀完這篇文章,希望小編分享的“Think-Swoole之WebSocket-Room加入、離開房間和房間消息發(fā)送的案例”這篇文章對大家有幫助,同時也希望大家多多支持億速云,關注億速云行業(yè)資訊頻道,更多相關知識等著你來學習!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI