网络编程
位置:首页>> 网络编程>> php编程>> swoole和websocket简单聊天室开发

swoole和websocket简单聊天室开发

作者:JohnsonChung  发布时间:2023-10-23 07:20:24 

标签: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下载

来源:http://www.cnblogs.com/johnson108178/archive/2017/11/18/7855924.html

0
投稿

猜你喜欢

  • 很多人都有研究闭合浮动元素的问题,但是解决方法却不一样,也并不是每一种方法都尽善尽美。闭合浮动元素(或者叫清除浮动)是web标准设计中经常会
  • 其实这里的静态页面并不是真正意义上的静态,但可以达到了静态页面的解析效率,还未经项目测试,拿来分享。代码如下:<% Cons
  • 本篇没有考虑异步,多线程及SQL注入WebDatabase 规范中说这份规范不再维护了,原因是同质化(几乎实现者都选择了Sqlite),且不
  • css可以处理16,777,216颜色,可以使用名字、rgb值或十六进制代码。red红色等同于 rgb(255,0,0) &nbs
  • ASP与MySQL的连接ASP和MySQL连接目前有两种方法:一种方法是使用MySQLX之类的组件,不过这种连接方法需要支付一定的费用;另外
  • 有很多种方法来实现图片的预加载,通常大部分使用Javascript让事情滚动。不要再受Javascript预载的束缚了吧,用CSS你就可以毫
  • 这个技巧将教你如何用css做出漂亮的文本按钮,有活力的按钮将节省你很多制作图片的时间,也能让你一天的工作中成为一个快乐的人,让我们一起看看效
  • 问题背景:日常对Sql Server 2005关系数据库进行操作时,有时对数据库(如:Sharepoint网站配置数据库名Sharepoin
  • 主要原理:调整dicom的窗宽,使之各个像素点上的灰度值缩放至[0,255]范围内。使用到的python库:SimpleITK下面是一个将d
  • 级联查询在ORACLE 数据库中有一种方法可以实现级联查询select *    //要查询的字段from table
  • 本文介绍了用ASP的AdoDb.Stream读取/写入UTF-8编码格式的文件的方法:函数名称:ReadTextFile 作用:利用AdoD
  • 前言近期在刷新生产环境数据库的时候,需要更新表中的字段,如果对每条数据结果都执行一次update语句,占用的数据库资源就会很多,而且速度慢。
  • 在本身比较复杂的页面里,再突出信息,往往是把几种方法叠加起来使用,比如加粗加大、加粗加色等,区别在于使用的类别和程度。导致的结果是呈现越来越
  • 选择了MySQL的安装版本后,要做的第二项决策是你是使用源码分发版还是二进制分发版。大多数情况,如果你的平台上已经有了一个二进制分发版,你可
  • 本文实例为大家分享了python可视化动态CPU性能监控的具体代码,供大家参考,具体内容如下打算开发web性能监控,以后会去学js,现在用m
  • MySQL中有很多的基本命令,show命令也是其中之一,在很多使用者中对show命令的使用还容易产生混淆,本文主要介绍了show命令的主要用
  • 利用ASP从远程服务器上接收XML数据<% dim objXML dim objRootElement dim&n
  • 在获得网页响应对象res后,使用res.text属性可以获得网页源代码,但可能出现乱码!因为requests库会使用自动猜测的解码方式将抓取
  • pytorch默认使用单精度float32训练模型,原因在于:使用float16训练模型,模型效果会有损失,而使用double(float6
  • 先看一下br怎么玩转“清除浮动”了。使用以下代码<br clear="all" />以下是代码效果演示:运行
手机版 网络编程 asp之家 www.aspxhome.com