用swoole和websocket開發簡單聊天室

首先,我想說下寫代碼的一些習慣,第一,任何可配置的參數或變量都要寫到一個config文件中。第二,代碼中必定要有日誌記錄和完善的報錯並記錄報錯。言歸正傳,swoole應該是每一個phper必需要了解的,它號稱從新定義了php。此聊天室利用了swoole高併發而且異步非阻塞的特色提升了程序的性能。php

首先,定義一個 swoole_lock 和 swoole_websocket_server ,而且配置參數,具體參數詳情能夠去swoole官網查看。git

public function start()
{ $
this->lock = new swoole_lock(SWOOLE_MUTEX);            // 對文件或數組進行鎖操做,已達到同步 $this->server = new swoole_websocket_server($this->addr, $this->port);  // swoole提供的Websocket Server $this->server->set(array( 'daemonize' => 0, 'worker_num' => 4, 'task_worker_num' => 10, 'max_request' => 1000, 'log_file' => ROOT_PATH . 'storage\\logs\\swoole.log' // swoole日誌路徑,必須是絕對路徑 )); $this->server->on('open', array($this, 'onOpen')); $this->server->on('message', array($this, 'onMessage')); $this->server->on('task', array($this, 'onTask')); $this->server->on('finish', array($this, 'onFinish')); $this->server->on('close', array($this, 'onClose'));
   // 啓動服務 $
this->server->start(); }

當有客戶端連接時,簡單記錄客戶端的信息。github

        public function onOpen($server, $request)
        {
            $message = array(
                'remote_addr' => $request->server['remote_addr'],
                'request_time' => date('Y-m-d H:i:s', $request->server['request_time'])
            );
            write_log($message);
        }

當有客戶端發送信息時,對信息進行處理。web

        public function onMessage($server, $frame)
        {
            $data = json_decode($frame->data);

            switch ($data->type) {
                case 'init':
                case 'INIT':
                    $this->users[$frame->fd] = $data->message;  // 記錄每一個連接的信息,一樣不要嘗試打印出來看,由於你只能看到本身的連接信息
                    $message = '歡迎' . $data->message . '加入了聊天室';
                    $response = array(
                        'type' => 1,    // 1表明系統消息,2表明用戶聊天
                        'message' => $message
                    );
                    break;
                case 'chat':
                case 'CHAT':
                    $message = $data->message;
                    $response = array(
                        'type' => 2,    // 1表明系統消息,2表明用戶聊天
                        'username' => $this->users[$frame->fd],
                        'message' => $message
                    );
                    break;
                default:
                    return false;
            }
                // 將信息交給task處理 $
this->server->task($response); } public function onTask($server, $task_id, $from_id, $message) {
       // 迭代全部的客戶端連接,將消息推送過去。(若是你嘗試將 $this->server->connections 打印出來,那麼你會發現他是空的。但當時用 foreach 去循環時,它確實有用。)
foreach ($this->server->connections as $fd) { $this->server->push($fd, json_encode($message)); } $server->finish( 'Task' . $task_id . 'Finished' . PHP_EOL); }

最後,當客戶端斷開連接時,利用鎖機制,同步刪除客戶端信息,並記錄日誌。json

        public function onClose($server, $fd)
        {
            $username = $this->users[$fd];
            // 釋放客戶端,利用鎖進行同步
            $this->lock->lock();
            unset($this->users[$fd]);
            $this->lock->unlock();

            if( $username ) {
                $response = array(
                    'type' => 1,    // 1表明系統消息,2表明用戶聊天
                    'message' => $username . '離開了聊天室'
                );
                $this->server->task($response);
            }


            write_log( $fd . ' disconnected');
        }

服務端完了,下面就是客戶端,很簡單,只須要用websocket連接就ok!數組

        // websocket
        let address = 'ws://<?php echo CLIENT_CONNECT_ADDR . ':' . CLIENT_CONNECT_PORT ?>';
        let webSocket = new WebSocket(address);
        webSocket.onerror = function (event) {
            alert('服務器鏈接錯誤,請稍後重試');
        };
        webSocket.onopen = function (event) {
            if(!sessionStorage.getItem('username')) {
                setName();
            }else {
                username = sessionStorage.getItem('username')
                webSocket.send(JSON.stringify({
                    'message': username,
                    'type': 'init'
                }));
            }
        };
        webSocket.onmessage = function (event) {
            console.log(event);
            let data = JSON.parse(event.data);
            if (data.type == 1) {
                $('#chat-list2').append('<li class="ui-border-tb"><span class="username">系統消息:</span><span class="message">' + data.message + '</span></li>');
            } else if (data.type == 2) {
                $('#chat-list2').append('<li class="ui-border-tb"><span class="username">' + data.username + ':</span><span class="message">' + data.message + '</span></li>');
            }

        };
        webSocket.onclose = function (event) {
            alert('散了吧,服務器都關了');
        };

 詳細代碼能夠去個人github下載服務器

相關文章
相關標籤/搜索