当前位置:首页 > PHP > 正文内容

php端口复用,php socket端口复用

高老师6年前 (2019-11-04)PHP15558

第一次听说端口复用是在mixphp最新版本中发现的,mixphp启动监听9501端口,现在作者说可以多开几个进程来执行mixphp,我心里想了下再启动不是会端口冲突嘛,但是却没有问题,于是下载mixphp的源码解读,原来是启动http服务器使用new Co\Http\Server('0.0.0.0', 80, false, true); swoole这个http构造方法第4个参数是是否开启端口复用,这样就能多个进程来监听相同端口。底层负载调度由linux自动处理。切记需要linux3.10以上内核才支持。我们来做个试验。

【一】.基于swoole的端口复用

(1).创建t1.php:监听80端口

go(function () {
    $server = new Co\Http\Server('0.0.0.0', 80, false, true);
    $server->handle('/', function ($request, $response) {
        $response->end("<h1>T1 is work!</h1>");
        echo 'replay from T1' . PHP_EOL;
    });
    $server->start();
});

(2).创建t2.php:监听80端口

go(function () {
    $server = new Co\Http\Server('0.0.0.0', 80, false, true);
    $server->handle('/', function ($request, $response) {
        $response->end("<h1>T2 is work!</h1>");
        echo 'replay from T2' . PHP_EOL;
    });
    $server->start();
});

(3).通过ab.exe并发请求:

./ab.exe -n 1000 -c 1000 http://127.0.0.1/

(4).查看服务器php脚本输出:

replay from T2
replay from T1
replay from T2
replay from T1
replay from T1
replay from T2
replay from T1
replay from T2
replay from T1

(5).关闭t1脚本继续访问80端口:

replay from T2
replay from T2
replay from T2

通过上面的验证我们可以知道端口复用能够提高多核利用率,提高socket的处理能力,同时也实现了高可用,一个进程意外退出,其他继续工作,不需要重新建立socket句柄。另外端口重用解决了端口占用不能及时释放的问题,不释放我们可以直接重用。

【二】.基于原生socket的端口复用

(1).创建w1.php:监听8484端口

// 设置IP
$host = '0.0.0.0';

// 设置端口
$port = 8484;

// 创建socket
$listen_socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);

// 设置端口复用
socket_set_option($listen_socket, SOL_SOCKET, SO_REUSEPORT, 1);

// 绑定端口
socket_bind($listen_socket, $host, $port);

// 监听socket
socket_listen($listen_socket);

// 事件轮询
while (true)
{
    // 客户端
    $client = socket_accept($listen_socket);

    // 向客户端发送消息
    $msg = "This is w1\r\n";

    // 记录响应来自哪个进程
    echo 'replay from W1' . PHP_EOL;

    //发送消息
    socket_write($client, $msg, strlen($msg));

    //关闭客户端
    socket_close($client);
}
socket_close($listen_socket);

(2).创建w2.php:监听8484端口

// 设置IP
$host = '0.0.0.0';

// 设置端口
$port = 8484;

// 创建socket
$listen_socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);

// 设置端口复用
socket_set_option($listen_socket, SOL_SOCKET, SO_REUSEPORT, 1);

// 绑定端口
socket_bind($listen_socket, $host, $port);

// 监听socket
socket_listen($listen_socket);

// 事件轮询
while (true)
{
    // 客户端
    $client = socket_accept($listen_socket);

    // 向客户端发送消息
    $msg = "This is w2\r\n";

    // 记录响应来自哪个进程
    echo 'replay from W2' . PHP_EOL;

    //发送消息
    socket_write($client, $msg, strlen($msg));

    //关闭客户端
    socket_close($client);
}
socket_close($listen_socket);

(3).并发后两个脚本的输出:

replay from W1
replay from W1
replay from W1
replay from W1
replay from W1
replay from W1
replay from W1
replay from W1
replay from W1
replay from W2
replay from W2
replay from W2
replay from W2
replay from W2
replay from W2
replay from W2
replay from W2
replay from W2
replay from W2
replay from W2

Linux底层会自动负载均衡。

注意事项:

linux内核必须大于等于3.10以上

windows下的php原生socket不支持端口复用。

windows下的linux子系统wsl1暂时对内核实现不完整,测试不出负载调度效果。

扫描二维码推送至手机访问。

版权声明:本文由高久峰个人博客发布,如需转载请注明出处。

本文链接:https://blog.20230611.cn/post/111.html

分享给朋友:

“php端口复用,php socket端口复用” 的相关文章

php抽奖概率算法

php抽奖概率算法

<?php /*  *算法学习自百度.只是学习和记录  */ header("Content-type:text/html;charset=utf-8"); //1.设置奖项,id是奖项id,name是中奖名称,v是中奖概率 $arr =&n...

php非对称加密

php非对称加密

先在centos安装openssl,然后开始://生成私钥openssl genrsa -out rsa_private_key.pem 1024//生成公钥openssl rsa -in rsa_private_key.pem&...

 php命令行中文乱码,php cli中文乱码

php命令行中文乱码,php cli中文乱码

<?php //如果支持exec函数,可以使用的方式 exec('chcp 65001'); //如果exec函数因安全问题禁用,可以使用的方式 pclose(popen('chcp 65001', 'r'));...

php下载远程文件(支持断点续传,支持超大文件)

php下载远程文件(支持断点续传,支持超大文件)

断点下载的原理:http请求头添加Range参数告诉文件服务器端需要的字节范围例如1个文本文件的字节为1000,第一次请求Range: bytes=0-500第二次请求Range: bytes=501-1000通过每次的请求将返回的流追加写入到文件。注意的项目:断点下载服务器端的每次只返回字节传输的...

php爬虫执行js,php执行js

php爬虫执行js,php执行js

当我们使用php爬虫采集网站时经常会遇到内容使用ajax异步加载。一般采取的方案是PHP模拟再请求api接口获取数据,但是有时候前端js加密非常麻烦,我们需要将js的加密方法转换为php方法方便curl请求。当然通过了解我们可以通过3种方案解决。第一种:使用phpv8js扩展执行js代码。(pecl...

php管道通信

php管道通信

(1).管道是干嘛的?管道是用于进程之间通信的,传播或交换信息(2).管道有几种?(2.1).匿名管道(pipe):管道是一种半双工的通信方式,数据只能单向流动,而且只能在具有亲缘关系的进程间使用。通常是指父子进程关系。(2.2).高级管道(popen):将另一个程序当做一个新的进程在当前程序进程中...