核心的swoole代码

基本的cs(client-sercer)结构不变,这里利用的是redis的哈希和set来储存和分组;从而达到了分组,统计,定时推送等功能;最后利用onclose事件来剔除断开的连接,全部代码如下:(没做前端,就不展示了)

核心的swoole ws.php

<?php 

namespace app\common;
require_once 'Predis.php';
require_once 'Task.php';
/**
* socket面向对象的编译
*/
class Ws
{
CONST HOST='0.0.0.0';
CONST PORT='9501';
public $ws=null;
public $getmsg=null;
public $server=null; public function __construct()
{
$this->ws=new \swoole_websocket_server(self::HOST,self::PORT);
$this->ws->set([
//启动task必须要设置其数量
'worker_num' => 4,
'task_worker_num' => 2,
// 'heartbeat_check_interval' => 5,
// 'heartbeat_idle_time' => 10,
]);
//监听新端口
$this->server=$this->ws->listen("127.0.0.1", 9502, SWOOLE_SOCK_TCP);
//关闭websocket模式
$this->server->set([
'open_websocket_protocol' => false,
]); $this->ws->on("start", [$this, 'onStart']);
$this->ws->on('open',[$this,'onopen']);
$this->server->on("receive", [$this, 'onReceive']);
$this->ws->on('task',[$this,'onTask']);
$this->ws->on('finish',[$this,'onFinish']);
$this->ws->on('message',[$this,'onmessage']);
$this->ws->on('close',[$this,'onclose']);
$this->server->on("close", [$this, 'oncloses']);
$this->ws->start();
}
//监听数据接收事件
public function onReceive($serv, $fd, $from_id, $data)
{
$shuju=json_decode($data,ture);
// print_r($shuju).PHP_EOL;
if (empty($shuju['data'])) {
$this->ws->push(Predis::getInstance()->get('fd'), $data);
}else{
if (empty($shuju['msg'])) {
//执行异步任务
$this->ws->task($shuju);
}else{
$push_arr=Predis::getInstance()->hvals($shuju['data']);
// echo "集群是:".print_r($push_arr);
foreach ($push_arr as $v) {
$this->ws->push($v, $shuju['msg']);
}
}
}
}
/**
* 设置进程名,为后续平滑重启进程
* @param $server
*/
public function onStart($server) {
swoole_set_process_name("live_master");
}
/**
监听开启事件的回调
*/
public function onopen($server, $request)
{
print_r("这时的fd是:",$request->fd);
Predis::getInstance()->set('fd',$request->fd);
} /**
监听接收事件的回调
*/
public function onmessage($server, $frame)
{
$server->push($frame->fd, "{$frame->data}");
}
/**
监听关闭事件的回调
*/
public function onclose($ser, $fd)
{
print_r("你好,我的{$fd}\n");
//退出并删除多余的分组fd
$group=Predis::getInstance()->sMembers('group');
foreach ($group as $v) {
$fangjian=Predis::getInstance()->hgetall($v);
foreach ($fangjian as $k => $vv) {
if ($fd == $vv) {
Predis::getInstance()->hdel($v,$k);
}
}
}
}
public function oncloses($ser, $fd)
{
print_r("这个是client{$fd}\n");
} /**
* $serv 服务
* $task_id 任务ID,由swoole扩展内自动生成,用于区分不同的任务
* $src_worker_id $task_id和$src_worker_id组合起来才是全局唯一的,不同的worker进程投递的任务ID可能会有相同
* $data 是任务的内容
*/
public function onTask($serv,$task_id,$src_worker_id,$data)
{
//引入任务
$obj = new Task;
$method = $data['data'];
$arr = $data['arr'];
//发布具体的任务
$flag = $obj->$method($arr, $serv);
return $flag; // 告诉worker
}
/**
* $task_id 是任务的ID
* $data 是任务处理的结果内容
*/
public function onFinish($serv,$task_id,$data)
{
print_r($data).'/n';
} } new Ws();

分发任务task.php

<?php
/**
* 代表的是 swoole里面 后续 所有 task异步 任务 都放这里来
* Date: 18/3/27
* Time: 上午1:20
*/
namespace app\common;
// include 'Predis.php'; class Task {
//异步创建房间
public function chuangjian($data,$serv)
{
$time=$data['time']*1000;
swoole_timer_after($time, function() use($data){
//创建房间(修改拍卖商品状态)
self::post("https://code.77wx.cn/index/index/in");
});
} //进入房间并缓存信息
public function jingru($data,$serv)
{
$fd=Predis::getInstance()->get('fd');
//加入分组
Predis::getInstance()->hset($data['name'],$data['uid'],$fd);
//加入组集合
Predis::getInstance()->sadd('group',$data['name']);
} public function post($url,$params=false,$ispost=0)
{
$httpInfo = array();
$ch = curl_init();
curl_setopt( $ch, CURLOPT_HTTP_VERSION , CURL_HTTP_VERSION_1_1 );
curl_setopt( $ch, CURLOPT_USERAGENT , 'Mozilla/5.0 (Windows NT 5.1) AppleWebKit/537.22 (KHTML, like Gecko) Chrome/25.0.1364.172 Safari/537.22' );
curl_setopt( $ch, CURLOPT_CONNECTTIMEOUT , 30 );
curl_setopt( $ch, CURLOPT_TIMEOUT , 30);
curl_setopt( $ch, CURLOPT_RETURNTRANSFER , true );
if( $ispost )
{
curl_setopt( $ch , CURLOPT_POST , true );
curl_setopt( $ch , CURLOPT_POSTFIELDS , $params );
curl_setopt( $ch , CURLOPT_URL , $url );
}
else
{
if($params){
curl_setopt( $ch , CURLOPT_URL , $url.'?'.$params );
}else{
curl_setopt( $ch , CURLOPT_URL , $url);
}
}
//执行
$response = curl_exec( $ch );
if ($response === FALSE) {
//echo "cURL Error: " . curl_error($ch);
return false;
} $httpCode = curl_getinfo( $ch , CURLINFO_HTTP_CODE );
$httpInfo = array_merge( $httpInfo , curl_getinfo( $ch ) );
//关闭url请求
curl_close( $ch );
return json_decode($response,1);
} }

客户端 client.php

<?php
namespace app\common; class Client
{
public $msg=''; public $data=[]; public function lianjie(){ $cli = new \swoole_client(SWOOLE_SOCK_TCP);
//判断连接状态(同步连接模式)
$res=$cli->connect('127.0.0.1', 9502);
if (empty($res)) {
return "连接失败";
} if (!empty($this->data)) {
//发送消息给server
$rel=$cli->send(json_encode($this->data));
}else{
//发送消息给server
$rel=$cli->send($this->msg);
}
if (!empty($rel)) {
return $rel;
}else{
return flash;
}
}
}

控制器index.php

<?php
namespace app\index\controller; use app\common\Client;
use app\common\Predis;
use app\common\Sql;
use app\index\model\User; class Index
{
//创建房间(添加拍卖倒计时)
public function chuangjian()
{
$data['time']=input("time");
$data['id']=input("id");
$cli = new Client();
$cli->data = [
'data' => 'chuangjian',
'arr' => $data
];
return $cli->lianjie();
}
//点击添加哈希(进入房间)
public function jingru()
{
$data['name']=input("name");
$data['uid']=input("uid");
$cli = new Client();
$cli->data = [
'data' => 'jingru',
'arr' => $data
];
return $cli->lianjie();
}
//本房间推送(出价格成功并推送)
public function pushfan()
{
$data['fan']=input("fan");
$cli = new Client();
$cli->data = [
'data' => $data['fan'],
'msg' => "恭喜用户111,喜当爹!!!!"
];
return $cli->lianjie();
}
//时间结束并指定推送
public function zhiding()
{
$data['fan']=input("fan");
$cli = new Client();
$cli->data = [
'data' => $data['fan'],
'msg' => "恭喜用户111,喜当爹!!!!"
];
return $cli->lianjie();
} }

完事!

基于swoole实现多人聊天室的更多相关文章

  1. 基于Socket实现多人聊天室

    当前支持: 1.仅文字 2.加入聊天室提醒 3.退出聊天室提醒 可能出现的BUG: 1.可能出现客户端发送信息后不能及时推送,需要下一个客户端发送信息后一起推送 服务端代码: 1 package co ...

  2. 与众不同 windows phone (31) - Communication(通信)之基于 Socket UDP 开发一个多人聊天室

    原文:与众不同 windows phone (31) - Communication(通信)之基于 Socket UDP 开发一个多人聊天室 [索引页][源码下载] 与众不同 windows phon ...

  3. 与众不同 windows phone (30) - Communication(通信)之基于 Socket TCP 开发一个多人聊天室

    原文:与众不同 windows phone (30) - Communication(通信)之基于 Socket TCP 开发一个多人聊天室 [索引页][源码下载] 与众不同 windows phon ...

  4. 基于tcp和多线程的多人聊天室-C语言

    之前在学习关于网络tcp和多线程的编程,学了知识以后不用一下总绝对心虚,于是就编写了一个基于tcp和多线程的多人聊天室. 具体的实现过程: 服务器端:绑定socket对象->设置监听数-> ...

  5. 多人聊天室(Java)

    第1部分 TCP和UDP TCP:是一种可靠地传输协议,是把消息按一个个小包传递并确认消息接收成功和正确才发送下一个包,速度相对于UDP慢,但是信息准确安全:常用于一般不要求速度和需要准确发送消息的场 ...

  6. Vue运用swoole 实现多人聊天

    1:项目目录新建php文件和html文件,这里我建了一个swoole文件夹,chat.php 和chat.html 2:将项目同步上线 0.0.0.0 代表的是所有人都可以连接. chat.php代码 ...

  7. Apache MiNa 实现多人聊天室

    Apache MiNa 实现多人聊天室 开发环境: System:Windows JavaSDK:1.6 IDE:eclipse.MyEclipse 6.6 开发依赖库: Jdk1.4+.mina-c ...

  8. 基于Server-Sent Event的简单聊天室 Web 2.0时代,即时通信已经成为必不可少的网站功能,那实现Web即时通信的机制有哪些呢?在这门项目课中我们将一一介绍。最后我们将会实现一个基于Server-Sent Event和Flask简单的在线聊天室。

    基于Server-Sent Event的简单聊天室 Web 2.0时代,即时通信已经成为必不可少的网站功能,那实现Web即时通信的机制有哪些呢?在这门项目课中我们将一一介绍.最后我们将会实现一个基于S ...

  9. 基于LINUX的多功能聊天室

    原文:基于LINUX的多功能聊天室 基于LINUX的多功能聊天室 其实这个项目在我电脑已经躺了多时,最初写完项目规划后,我就认认真真地去实现了它,后来拿着这个项目区参加了面试,同样面试官也拿这个项目来 ...

随机推荐

  1. Delphi xe 10.3.2-快递接口封装-【快递鸟(即时查询和单号识别)】

    编译环境:Windows 7 +Delphi xe 10.3.2 封装了快递鸟接口,注意的坑:MD5要转为小写. function TKDniaoAPI.StrtoMd5(const str: str ...

  2. MySQL 在 Windows 下安装教程、避坑指南

    MySQL 是一个关系型数据库管理系统,由瑞典 MySQL AB 公司开发,2008 年被 SUN 公司收购,后 SUN 公司又被 Oracle 公司收购. 一.下载 MySQL 官网 https:/ ...

  3. SpringCloud微服务(01):Eureka组件,管理服务注册与发现

    本文源码:GitHub·点这里 || GitEE·点这里 一.Eureka基本架构 1.Eureka简介 Eureka是Netflix开发的服务发现框架,本身是一个基于REST的服务,SpringCl ...

  4. 《少年的你》票房被刷爆?让我用python分析一波它好看在哪里!

    最近少年的你刷爆票房,但是是真好看还是假好看,我们也不知道,所以让我们用python来分析一下~ 票房过12亿 两位主演粉丝加起来过亿 电影话题量过亿 豆瓣8.4分, 时光网8.4分, IMDB 7. ...

  5. 大数据相关概念和hdfs

    大数据 概述 大数据是新处理模式才能具备更多的决策力,洞察力,流程优化能力,来适应海量高增长率,多样化的数据资产. 大数据面临的问题 怎么存储海量数据(kb,mb,gb,tb,pb,eb,zb) 怎么 ...

  6. scrapy实例:爬取天气、气温等

    1.创建项目 scrapy startproject weather # weather是项目名称 scrapy crawl spidername开始运行,程序自动使用start_urls构造Requ ...

  7. 教你两招用纯CSS写Tab切换

    说到Tab切换,你可能首先想到的就是使用jQuery,短短几行代码就可以轻松搞定一个Tab切换. 而今天所要分享的,是使用 0 行JS代码来实现Tab切换! 具体效果如下:   Tab切换 方法一:模 ...

  8. Linux:系统的启动过程

    Linux系统的启动过程 过程 通电-> BIOS-> LILO/GRUB-> Kernel Boot-> init->rc.sysinit init->rc -& ...

  9. STL 中 list 的使用

    list 容器实现了双向链表的数据结构,数据元素是通过链表指针串连成逻辑意义上的线性表,这样,对链表的任一位置的元素进行插入.删除和查找都是极快速的.由于list对象的节点并不要求在一段连续的内存中, ...

  10. vue模板语法下

    样式绑定 <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <titl ...