利用Swoole编写一个TCP服务器,顺带测试下Swoole的4层生命周期
1首先我们写一个入口脚本,这里简单点的功能就是开启服务和关闭服务
<?php
//CLI命令
if(isset($argv[1]) && in_array($argv[1], ['start', 'restart', 'status', 'stop', 'reload']))define('CLI_COMMAND', $argv[1]);
else define('CLI_COMMAND', 'start');
define('ROOT',__DIR__);
define('TMP_PATH',ROOT.DIRECTORY_SEPARATOR.'tmp'.DIRECTORY_SEPARATOR);
define('APP_NAME','mySwoole');
require_once 'ServerCallFun.php';
Class Root
{
static public $Ip = '127.0.0.1';
static public $port = 9510;
//swoole对象
Static Public $serv = null; //进程对象
Static Public $worker = null; /**
* 主框架运行
*/
Static Public function run()
{
$command = 'Root::' . CLI_COMMAND;
$command();
} /**
* 启动框架
*/
Static Private function start()
{
if(is_file(TMP_PATH . 'server.pid')){
$pid = @file_get_contents(TMP_PATH . 'server.pid');
if($pid && \swoole_process::kill($pid, 0))die("Framework has been started!" . PHP_EOL);
}
echo "Framework Starting...", PHP_EOL;
date_default_timezone_set('Asia/Shanghai');
$setup = [
'pid_file' => TMP_PATH.'server.pid',//在Server启动时自动将master进程的PID写入到文件,在Server关闭时自动删除PID文件
'reactor_num' => 4,//reactor_num建议设置为CPU核数的1-4倍
'worker_num' => 4,
'backlog' => 128,//Listen队列长度,如backlog => 128,此参数将决定最多同时有多少个等待accept的连接。
'max_request' => 100,//设置worker进程的最大任务数,默认为0,一个worker进程在处理完超过此数值的任务后将自动退出,进程退出后会释放所有内存和资源。
'dispatch_mode' =>3,//抢占模式,主进程会根据Worker的忙闲状态选择投递,只会投递给处于闲置状态的Worker
'daemonize'=>true,
'log_file' => TMP_PATH .'swoole.log',//开启守护进程模式后(daemonize => true),标准输出将会被重定向到log_file。在PHP代码中echo/var_dump/print等打印到屏幕的内容会写入到log_file文件日志标号 ];
self::$serv = new swoole_server(self::$Ip,self::$port) or die('Swoole Starting Failed!' . PHP_EOL);
self::$serv->set($setup);
self::$serv->on('start', 'ServerCallFun::start');//在主进程start后调用
self::$serv->on('managerstart', 'ServerCallFun::managerStart');//管理进程开启后 //设置工作/任务进程启动回调
self::$serv->on('workerstart', 'ServerCallFun::onWorkstart');
/*self::$serv->on('request',function ($request, $response) {
$response->header("Content-Type", "text/html; charset=utf-8");
$response->end("<h1>Hello Swoole. #".rand(1000, 9999)."</h1>");
});*/
//监听连接进入事件
self::$serv->on('connect',function ($serv, $fd) {
echo "Client: Connect.\n";
}); //监听数据接收事件
self::$serv->on('receive','ServerCallFun::onReceive'); //监听连接关闭事件
self::$serv->on('close',function ($serv, $fd) {
echo "Client: Close.\n";
}); //在启动之前添加一个全局的内存tabel监控所有的进程状态 //实例启动前执行(start前的状态全部是程序全局期)
self::$serv->start();
} Static Private function stop(){
$pid = @file_get_contents(TMP_PATH . 'server.pid');
if($pid){
if(\swoole_process::kill($pid, 0))\swoole_process::kill($pid, 15);
else{
foreach(glob(TMP_PATH . '*.pid') as $filename){
$pid = @file_get_contents($filename);
if(\swoole_process::kill($pid, 0))\swoole_process::kill($pid, 9);
@unlink($filename);
}
}
die('Stop of Framework Success!' . PHP_EOL);
}
die('Framework not started!' . PHP_EOL);
} }
Root::run();
进一步来了解下这里是主要用来干嘛的
首先是声明了当前的执行文件的绝对路径,这里pid_file一定是要绝对路径
这里运行start命令会调用start的方法,这里主要是设置配置参数然后开启swoole默认的tcp服务。swoole的回调函数我放到另外一个类中处理了,这里是为了理解swoole的生命周期有意为之。
设置的默认参数中比较重要的是pid_file这个参数,这个可以用来重启和关闭服务进程。
在start方法执行前,所有的require和变量都是在第一层swoole生命周期中为程序全局期。
回调函数类是
<?php
require_once 'CmdController.php';
class ServerCallFun{
static function start(\swoole_server $server){
\swoole_set_process_name("Master process");
//绑定状态事件
//\swoole_process::signal(SIGUSR1, '\Root\Http::status');
//绑定重载事件
\swoole_process::signal(SIGUSR2, function() use ($server){
$server->reload();
});
} static function managerStart(\swoole_server $server){
$num = count(glob(TMP_PATH . 'manager_*.pid'));
file_put_contents(TMP_PATH . 'manager_'. $num .'.pid', $server->manager_pid);
\swoole_set_process_name("Manager[{$num}] process");
} static Public function onWorkstart(\swoole_server $server, int $worker_id)
{
//实例化进程对象
Root::$worker = new self();
if($server->taskworker) {
file_put_contents(TMP_PATH . "task_{$worker_id}.pid", $server->worker_pid);
\swoole_set_process_name("Tasker[{$worker_id}] process ");
echo "TaskID[{$worker_id}] PID[". $server->worker_pid ."] creation finish!" . PHP_EOL;
} else {
file_put_contents(TMP_PATH . "worker_{$worker_id}.pid", $server->worker_pid);
\swoole_set_process_name("Worker[{$worker_id}] process");
echo "WorkerID[{$worker_id}] PID[". $server->worker_pid ."] creation finish!" . PHP_EOL;
}
} static Public function onConnect($serv, $fd)
{ } static Public function onReceive($serv, $fd, $from_id, $data)
{
$msg = CmdController::run($data,$fd,$from_id);
$serv->send($fd,json_encode($msg));
} static Public function onClose($serv, $fd)
{ }
}
这里我们在主进程开启后执行了start的回调,添加上了重启热更新子进程的信号siguser2.进行server reload.并且将进程名字设置为Master Process
对管理进程也设置了回调。也是为了更名
对work进程设置回调更名,同时因为是守护方式运行服务,php的输出会输出到log_file配置的路径中
然后我在测试receive中测试下进程全局期中会出现什么情况
<?php
class CmdController{
static public $count = 0;
static public function run($recive,$fd,$fromId){
self::$count++;
$returnArr = ['ret'=>-1,'data'=>'','msg'=>''];
if($recive){
$returnArr['ret'] = 0;
$returnArr['data'] = self::$count.";{$fd}-{$fromId}:getRequest[{$recive}]->getReponse[Success]";
}
return $returnArr;
}
}
如果是我们的理解,我们会认为每次tcp客户端发送数据,然后服务端接收到的话,就是让count+1。其实这里就是进程的全局期,4个work会独立保存各自的数据。所以会出现下面,count出现4次重复然后自增的情况。
如何开启测试的服务器 php index.php start
如何关闭测试的服务器 php index.php stop
利用Swoole编写一个TCP服务器,顺带测试下Swoole的4层生命周期的更多相关文章
- 如何编写一个 SendFile 服务器
如何编写一个 SendFile 服务器 前言 之前讨论零拷贝的时候,我们知道,两台机器之间传输文件,最快的方式就是 send file,众所周知,在 Java 中,该技术对应的则是 FileChann ...
- Swoole学习(七)Swoole之异步TCP服务器的创建
环境:Centos6.4,PHP环境:PHP7 <?php //创建TCP服务器 /** * $host 是swoole需要监听的ip,如果要监听本地,不对外服务,那么就是127.0.0.1;如 ...
- 使用.net core在Ubuntu构建一个TCP服务器
介绍和背景 TCP编程是网络编程领域最有趣的部分之一.在Ubuntu环境中,我喜欢使用.NET Core进行TCP编程,并使用本机Ubuntu脚本与TCP服务器进行通信.以前,我在.NET框架本身写了 ...
- 利用javafx编写一个时钟制作程序
1.首先创建一个时钟类,用于编写时钟的各种特有属性 package javaclock; /** * * @author admin */import java.util.Calendar;impor ...
- 27、通过visual s'tudio 验证 SOCKET编程:搭建一个TCP服务器
本文就是在windows下进行socket编程,搭建一个TCP客户端. 在visual studio下编程,首先在windows下进行初始化(这点在linux下是不需要的): /* 初始化 Winso ...
- 测试微信小程序页面的生命周期
前言:本人是一个初学者,也是第一次写博客,敲键盘的时候还不知道发布后是什么效果,希望内容给其他初学的同学一点帮助,同时加深自己的理解.这篇随笔讲的是Page页面的生命周期,在开发中是基础中的基础,很容 ...
- 如何利用tomcat搭建一个动态服务器
这篇文章只记录已解压缩包的方式安装,通常linux服务器上也是这样. 1.下载tomcat.zip压缩包. http://tomcat.apache.org/download-70.cgi 2.把zi ...
- 01.swoole学习笔记--TCP服务器
1.安装swoole扩展 2.网络调试助手进行调试 <?php //创建服务器 $host='192.168.10.31'; $port=; //$model='SWOOLE_PROCESS'; ...
- 13.首先,编写一个类ChongZai,该类中有3个重载的方法void print();其次, 再编写一个主类来测试ChongZai类的功能。
package java1; //计算器 public class Jisuanqi { //属性 //型号,品牌等 //重载 //1.方法同名不同参 //2.返回类型和重载无关 //3.多态的一种表 ...
随机推荐
- can协议
Controller Area Network,是一种用于实时应用的串行通讯协议总线. CAN控制器通过组成总线的2根线(CAN-H和CAN-L)的电位差来确定总线的电平,在任一时刻,总线上有2种电平 ...
- FMS Dev Guide学习笔记(验证客户端)
一.开发交互式的媒体应用程序 1.使用客户端对象的属性 当一个客户端连接上服务器上的一个应用,服务端就会创建一个包含这个客户端信息的客户端对象并且将它传递给application.onConn ...
- ubuntu 下 rvm 卸载和重装
卸载: sudo apt-get --purge remove ruby-rvm sudo rm -rf /usr/share/ruby-rvm /etc/rvmrc /etc/profile.d/r ...
- hive sql 效率提升
转 : http://www.cnblogs.com/xd502djj/p/3799432.html hive的查询注意事项以及优化总结 . Hive是将符合SQL语法的字符串解析生成可以在Hado ...
- Javascript概念
什么是JavaScript? 1 以前的目的:验证表单输入的正确性. 2 现在的目的:多做一些页面的交互效果. 3 javascript是一个跨平台的脚本语言. 4 网景公司开发的,由布兰登·艾奇最先 ...
- bootstrap 固定表头
1 htmL <!DOCTYPE html> <html> <head> <title>Fixed Columns</title> < ...
- War3编辑器
[War3编辑器] 1.英雄在商店买东西叫“任意单位 出售物品”,英雄卖东西给商店叫“抵押物品”. 2.触发器中的一级窗口包含: 1)设置事件窗口. 2)设置条件窗口. 3)设置动作窗口. 3.二级设 ...
- 解题(MiGong--迷宫问题(深度搜索))
题目描述 定义一个二维数组N*M(其中2<=N<=10;2<=M<=10),如5 × 5数组下所示: int maze[5][5] = { 0, 1, 0, 0, 0, 0, ...
- JMeter学习(十三)目录介绍(转载)
转载自 http://www.cnblogs.com/yangxia-test JMeter也学了一阵子了,对于基本的操作已了解,再回过头来看看Jmeter的目录,本篇是对于它的目录进行一些简单的介绍 ...
- Oracle 入门
一.安装Oracle 11g 服务端 服务端安装教程:https://jingyan.baidu.com/article/363872eccfb9266e4aa16f5d.html 二.安装客户端 客 ...