核心的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. pixijs shader 贴图溶解效果教程

    pixijs shader 贴图溶解效果教程 我直接贴代码了 没什么好讲解了 稍微有点基础的人应该能看懂 const app = new PIXI.Application({ transparent: ...

  2. Jrebel实现tomcat热部署,遇到的问题以及解决办法,详解

    我的安装的详细过程: 下载Jrebel:  https://github.com/ilanyu/ReverseProxy/releases/tag/v1.4 我的是winx64,所以选择如下的: 下载 ...

  3. Docker 零碎

    Delete none tag docker image: $ docker stop $(docker ps -a | grep "Exited" | awk '{print $ ...

  4. spring的简易实现(一)

    [练习]spring的简易实现(一) 在第一部分我们实现读取xml的配置,然后实例化xml中的bean 首先定义一个xml和相关的class类 <?xml version="1.0&q ...

  5. Monday

    ssm项目 设计模式:上周(观察者模式,策略模式)

  6. Java操作数据库——手动实现数据库连接池

    Java操作数据库——手动实现数据库连接池 摘要:本文主要学习了如何手动实现一个数据库连接池,以及在这基础上的一些改进. 部分内容来自以下博客: https://blog.csdn.net/soonf ...

  7. Java生鲜电商平台-商城优惠券设计要点复盘与总结

    Java生鲜电商平台-商城优惠券设计要点复盘与总结 Java生鲜电商平台本文将从优惠券设计用户端,需求端,业务流程全方案解析优惠券设计方案 为什么要设计优惠券 设计优惠券的核心:拉新和促活 新产品上线 ...

  8. Implement Dependent Reference Properties 实现从属引用属性 (XPO)

    In this lesson, you will learn how to implement properties whose values can depend on other properti ...

  9. axios请求无法携带cookie

    背景 最近在用vue重构公司公司AngularJS 1.X项目,老项目身份认证采用的cookie,前端ajax库采用是angularJS的$http/$resource服务,新版采用的是axios,但 ...

  10. 【React Native】进阶指南之二(手势响应系统)

    移动设备上的手势识别要比在 web 上复杂得多.用户的一次触摸操作的真实意图是什么,App 要经过好几个阶段才能判断.比如 App 需要判断用户的触摸到底是在滚动页面,还是滑动一个 widget,或者 ...