用swoole和websocket开发简单聊天室

首先,我想说下写代码的一些习惯,第一,任何可配置的参数或变量都要写到一个config文件中。第二,代码中一定要有日志记录和完善的报错并记录报错。言归正传,swoole应该是每个phper必须要了解的,它号称重新定义了php。此聊天室利用了swoole高并发并且异步非阻塞的特点提高了程序的性能。

首先,定义一个 swoole_lock 和 swoole_websocket_server ,并且配置参数,具体参数详情可以去swoole官网查看。

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();
}

当有客户端链接时,简单记录客户端的信息。

        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);
        }

当有客户端发送信息时,对信息进行处理。

        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);
        }

最后,当客户端断开链接时,利用锁机制,同步删除客户端信息,并记录日志。

        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下载

时间: 2024-10-28 14:07:11

用swoole和websocket开发简单聊天室的相关文章

Java和WebSocket开发网页聊天室

小编心语:咳咳咳,今天又是聊天室,到现在为止小编已经分享了不下两个了,这一次跟之前的又不大相同,这一次是网页聊天室,具体怎么着,还请各位看官往下看~ Java和WebSocket开发网页聊天室 一.项目简介 WebSocket是HTML5一种新的协议,它实现了浏览器与服务器全双工通信,这里就将使用WebSocket来开发网页聊天室,前端框架会使用AmazeUI,后台使用Java,编辑器使用UMEditor. 二.涉及知识点 网页前端(HTML+CSS+JS)和Java 三.软件环境 Tomcat

Flask基于websocket的简单聊天室

1.安装gevent-websocket pip install -i https://pypi.tuna.tsinghua.edu.cn/simple/ gevent-websocket 2.chat.py文件内容如下: from flask import Flask,request,render_template import json from geventwebsocket.handler import WebSocketHandler from gevent.pywsgi import

使用WebSocket实现网页聊天室

知道WebSocket挺久了,但是一直没提起兴趣去了解它. 今天听@成熟的毛毛虫 说到slack.小小的试用了一下,发现slack的聊天功能做得相当强大,看了下网络请求,发现是基于WebSocket实现的.顿时提起兴趣,想了解下这强大的WebSocket. 先了解下WebSocket. 开源中国的介绍:http://www.oschina.net/p/websocket 百度百科的介绍:http://baike.baidu.com/link?url=yMTBVobtkp1E1a0wrRJlnEX

分享基于 websocket 网页端聊天室

博客地址:https://ainyi.com/#/67 有一个月没有写博客了,也是因为年前需求多.回家过春节的原因,现在返回北京的第二天,想想,应该也要分享技术专题的博客了!! 主题 基于 websocket 网页端聊天室 WebSocket 协议是基于 TCP 的一种新的网络协议.它实现了浏览器与服务器全双工 (full-duplex) 通信--允许服务器主动发送信息给客户端. 使用 java 开发后台 需要导入一个jar包:javax.websocket-api-1.0-rc4.jar 后台

Python Socket 简单聊天室2

上篇文章写了一个简单的单线程的一问一答的简单聊天室.这次我们使用SocketServer模块搭建一个多线程异步的聊天室. 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 # -*- coding:utf-8 -*- import SocketServer class  mysocketclass(SocketServer.BaseRequestHandler):     def handle(self):         client_inform

ASP.NET 使用application和session对象写的简单聊天室程序

ASP.Net中有两个重要的对象,一个是application对象,一个是session对象. Application:记录应用程序参数的对象,该对象用于共享应用程序级信息. Session:记录浏览器端的变量对象,用来存储跨网页程序程序的变量或者对象. 说实话,写了快一年的asp.net,application对象还真没怎么用过.看了看书,根据这两个对象的特性写了一个简单的聊天室程序.真的是非常的简陋. 我的思路是,有两个页面Default页和ChatRoom页,页面布局如图: Default

C#实例之简单聊天室(状态管理)

前言        状态管理是在同一页或不同页的多个请求发生时,维护状态和页信息的过程.因为Web应用程序的通信协议使用了无状态的HTTP协议,所以当客户端请求页面时,ASP.NET服务器端都会重新生成一个网页实例.此时,旧网页的任务完成,旧网页的实例也随之消失.这种无状态,意味着客户端用户在浏览器中的一些状态或是对数据的一些修改都将丢失. 为了弥补这种基于web应用程序的固有限制,ASP.NET提供了多种用于管理状态的功能. 简单聊天室 这里运用System.Web命名空间的那些管理状态的类,

基于html5 localStorage , web SQL, websocket的简单聊天程序

new function() { var ws = null; var connected = false; var serverUrl; var connectionStatus; var sendMessage; var connectButton; var disconnectButton; var sendButton; var open = function() { var url = serverUrl.val(); ws = new WebSocket(url); ws.onope

Android 基于XMPP Smack openfire 开发的聊天室

Android基于XMPP Smack openfire 开发的聊天室