1.Workerman是什么?(套用官网)
Workerman是一款纯PHP开发的开源高性能的PHP socket 服务框架。
Workerman不是重复造轮子,它不是一个MVC框架,而是一个更底层更通用的socket服务框架,你可以用它开发tcp代理、梯子代理、做游戏服务器、邮件服务器、ftp服务器、甚至开发一个php版本的redis、php版本的数据库、php版本的nginx、php版本的php-fpm等等。Workerman可以说是PHP领域的一次创新,让开发者彻底摆脱了PHP只能做WEB的束缚。
实际上Workerman类似一个PHP版本的nginx,核心也是多进程+Epoll+非阻塞IO。Workerman每个进程能维持上万并发连接。由于本身常住内存,不依赖Apache、nginx、php-fpm这些容器,拥有超高的性能。同时支持TCP、UDP、UNIXSOCKET,支持长连接,支持Websocket、HTTP、WSS、HTTPS等通讯协以及各种自定义协议。拥有定时器、异步socket客户端、异步Mysql、异步Redis、异步Http、异步消息队列等众多高性能组件。
2. GatewayWorker是什么?(套用官网)
GatewayWorker基于Workerman开发的一个项目框架,用于快速开发TCP长连接应用,例如app推送服务端、即时IM服务端、游戏服务端、物联网、智能家居等等
GatewayWorker使用经典的Gateway和Worker进程模型。Gateway进程负责维持客户端连接,并转发客户端的数据给BusinessWorker进程处理,BusinessWorker进程负责处理实际的业务逻辑(默认调用Events.php处理业务),并将结果推送给对应的客户端。Gateway服务和BusinessWorker服务可以分开部署在不同的服务器上,实现分布式集群。
3. Gatewayworker + thinkphp
数据交互模型:
流程:
- 客户端(浏览器)发出socket请求与getwayworker建立连接
- 客户端发出http请求(注意是发送http请求处理业务,所有业务都放在tp处理)
- tp处理业务逻辑(把client_id与uid绑定、客户端分组、数据库查询等),然后调用gateway的接口把结果数据进行广播
- 客户端接收广播的数据,进行视图渲染
4. 环境搭建
1. workerman 与 gateway安装
composer require workerman/workerman
composer require workerman/gateway-worker
官网实例 点击下载
2. 配置
start_gateway.php文件配置(开启gateway服务,并在注册服务注册)
<?php
use \Workerman\Worker;
use \Workerman\WebServer;
use \GatewayWorker\Gateway;
use \GatewayWorker\BusinessWorker;
use \Workerman\Autoloader;
//
$context = array(
'ssl' => array(
'local_cert' => '/etc/pki/tls/certs/public.pem', // 或者crt文件
'local_pk' => '/etc/pki/tls/private/214498534070135.key',
'verify_peer' => false
)
);
// gateway 进程,这里使用websocket协议,这里使用了443端口,所有要加载ssl的配置,websocket://0.0.0.0:443:允许所有任何客户端使用wss协议访问
$gateway = new Gateway("websocket://0.0.0.0:443",$context);
$gateway->transport = 'ssl';
// gateway名称,status方便查看
$gateway->name = 'YourAppGateway';
// gateway进程数
$gateway->count = 4;
// 本机ip,分布式部署时使用内网ip
$gateway->lanIp = '172.31.240.231';
// 内部通讯起始端口,假如$gateway->count=4,起始端口为4000
// 则一般会使用4000 4001 4002 4003 4个端口作为内部通讯端口
$gateway->startPort = 2900;
// 服务注册地址
$gateway->registerAddress = '172.31.240.231:1238';
// 心跳检测 15秒一次
$gateway->pingInterval = 15;
$gateway->pingNotResponseLimit = 1;
// 当pingData为空,服务器将不会向客户端发送心跳检测(为了节省服务器资源,心跳检测最好由客户端发起)
$gateway->pingData = '';
3. start_businessworker.php(开启businessworker服务,并在注册服务注册)
<?php
use \Workerman\Worker;
use \Workerman\WebServer;
use \GatewayWorker\Gateway;
use \GatewayWorker\BusinessWorker;
use \Workerman\Autoloader;
// bussinessWorker 进程
$worker = new BusinessWorker();
// worker名称
$worker->name = 'YourAppBusinessWorker';
// bussinessWorker进程数量
$worker->count = 4;
// 服务注册地址
$worker->registerAddress = '172.40.239.231:1238';
// 如果不是在根目录启动,则运行runAll方法
if(!defined('GLOBAL_START')) {
Worker::runAll();
}
4. start_register.php(开启注册服务)
<?php
use \Workerman\Worker;
use \GatewayWorker\Register;
// register 服务必须是text协议
$register = new Register('text://172.40.239.231:1238');
// 如果不是在根目录启动,则运行runAll方法
if(!defined('GLOBAL_START')) {
Worker::runAll();
}
5. 启动服务
启动
以debug(调试)方式启动
php start.php start
以daemon(守护进程)方式启动
php start.php start -d
停止
php start.php stop
重启
php start.php restart
平滑重启
php start.php reload
查看状态
php start.php status
5. 客户端操作(这是我基于小程序接口,封装的一个socket库)
var socket = new Socket('wss://wss.xinyuruiyang.com');
socket.on("open",function (res) {
console.log('WebSocket连接已打开!')
_this.setData({
isOpenSocket: true
});
});
6. gateway端操作
<?php
use \GatewayWorker\Lib\Gateway;
use \Workerman\Lib\Timer;
class Events
{
public static $worker_id = null;
/**
* 当客户端连接时触发
* 如果业务不需此回调可以删除onConnect
*
* @param int $client_id 连接id
*/
public static function onConnect($client_id) {
// 向当前client_id发送数据(触发客户端的init时间)
Gateway::sendToClient($client_id, json_encode(["init",["client_id"=>$client_id]]));
}
}
7. tp端操作
private function bind_user($room,$client_id,$uid=''){
if( empty($uid) ){
$sk = $this->checksession();
$openid = $this->get_openid($sk);
$uid = D('User')->where(['openid'=>$openid])->getField('id');
}
// client_id与uid绑定
Gateway::bindUid($client_id, $uid);
// 加入某个群组(可调用多次加入多个群组)
Gateway::joinGroup($client_id, $room);
// 设置session
Gateway::setSession($client_id, ['uid'=>$uid]);
// 获取客户端分组人数
$number = Gateway::getClientCountByGroup($room);
D('Match')->where(['rid'=>$room,'uid'=>$uid,'status'=>['in',['1','2']]])->save(['client_id'=>$client_id]);
$message = D('Match')->where(['rid'=>$room,'status'=>['in',['1','2']]])->select();
foreach($message as &$v){
$v['user'] = D('User')->where(['id'=>$v['uid']])->find();
$v['user']['teamside'] = D('Match')->where(['rid'=>$room,'uid'=>$v['uid'],'status'=>['in',['1','2']]])->getField('teamside');
}
// 向某个房间广播数据
Gateway::sendToGroup($room, json_encode(["room",$message]));
}
}
来源:oschina
链接:https://my.oschina.net/u/4381939/blog/3855842