当前位置: 首页 > article >正文

Swoole的多进程模块

0ca5d202309271138275786.png

Swoole是有自己的一个进程管理模块,用来替代PHP的pcntl扩展,需要注意Process进程在系统是非常昂贵的资源,创建进程消耗很大,另外创建的进程过多会导致进程切换开销大幅上升。

为什么不使用pcntl

1.pcntl没有提供进程间通信的功能

2.pcntl不支持重定向标准输入和输出

3.pcntl只提供了fork这样原始的接口,容易使用错误

Swoole是怎么解决的

1.swoole_process提供了基于unixsock的进程间通信,使用很简单只需调用write/read或者push/pop即可

2.swoole_process支持重定向标准输入和输出,在子进程内echo不会打印屏幕,而是写入管道,读键盘输入可以重定向为管道读取数据

3.swoole_process提供了exec接口,创建的进程可以执行其他程序,与原PHP父进程之间可以方便的通信

创建进程

函数原型:

Swoole\Process::__construct(callable $function, $redirect_stdin_stdout = false, $create_pipe = true)

Copy

如果我们打印$process会发现,每次创建一个进程后,就会随之创建一个管道,主进程想和哪一个进程通信,就向那个进程的管道写入/读取数据。

管道有2个方法,分别来写入数据,和读取数据。

  1. $process->write(‘数据’);#写入数据
  2. $process->read()#读取数据

管道通讯方式一:

$worker = [];
for ($i = 0; $i < 3; $i++) {
    $process = new Swoole\Process(function ($process) {
        var_dump('子进程:' . $process->read());
        sleep(1);
        $process->write('子进程数据');
        echo PHP_EOL . posix_getpid() . PHP_EOL;
    }, false, true);
    $pid = $process->start();
    $worker[$pid] = $process;//把相应的进程放到同一个数组当中
    $process->write('主进程数据');
//    var_dump($process->read());//同步阻塞
}
foreach ($worker as $w) {
    var_dump('主进程:' . $w->read());
}

Copy

管道通讯方式二:

for ($i = 0; $i < 3; $i++) {
    $process = new Swoole\Process(function ($process) {
        var_dump('子进程:' . $process->read());
        sleep(1);
        $process->write('子进程数据');
        echo PHP_EOL . posix_getpid() . PHP_EOL;
    }, false, true);
    $pid = $process->start();
    $process->write('主进程数据');
    // 异步监听管道中的数据,读事件监听,当管道可读时触发
    swoole_event_add($process->pipe, function ($pipe) use ($process) {
        var_dump('主进程:' . $process->read());
    });
//    var_dump($process->read());//同步阻塞
}

Copy

消息队列的通讯

消息队列:

  1. 一系列保存在内核中的消息链表
  2. 有一个 msgKey, 可以通过此访问不同的消息队列
  3. 有数据大小限制, 默认 8192
  4. 阻塞 vs 非阻塞: 阻塞模式下 pop()空消息队列/push()满消息队列会阻塞, 非阻塞模式可以直接返回

swoole 中使用消息队列:

  1. 通信模式: 默认为争抢模式, 无法将消息投递给指定子进程
  2. 新建消息队列后, 主进程就可以使用
  3. 消息队列不可和管道一起使用, 也无法使用 swoole     event loop

步骤:

1.启用消息队列作为进程间通信:

bool swoole_process->useQueue(int $msgkey = 0, int $mode = 2);

Copy

2.投递数据到消息队列中:

 bool swoole_process->push(string $data);

Copy

3.从队列中提取数据

string swoole_process->pop(int $maxsize = 8192);

Copy

案例:

for ($i = 0; $i < 3; $i++) {
    $process = new Swoole\Process(function ($process) {
        var_dump('子进程:' . $process->pop());
//        $process->push('hello 主进程');#推送到主进程
    });
    $process->useQueue(1, 2 | swoole_process::IPC_NOWAIT);//启用消息队列,争抢模式,非阻塞,可能会被任意一个子进程接收到
    $pid = $process->start();


    $process->push('hello 子进程');#推送到子进程,不能当做管道使用
//    echo '主进程消息:' . $process->pop() . PHP_EOL;
}

Copy

常见问题


http://www.kler.cn/a/153910.html

相关文章:

  • Excel——宏教程(2)
  • 纯前端实现语音文字互转
  • ### 哋它亢在5G基站中的应用:新兴技术与未来通信的融合
  • React Native 全栈开发实战班 - 打包发布之热更新
  • Flutter在MaterialApp中的builder初始化多个包
  • Django5 2024全栈开发指南(三):数据库模型与ORM操作
  • 22、DS1302实时时钟
  • YOLOv8 第Y7周 水果识别
  • NXP iMX8M Plus Qt5 双屏显示
  • 【开源】基于JAVA语言的校园电商物流云平台
  • 匿名结构体类型、结构体的自引用、结构体的内存对齐以及结构体传参
  • Git多库多账号本地SSH连接配置方法
  • LINUX 嵌入式C编程--信号编程
  • IDEA在左下角显示当前代码所属的类名/方法名
  • vue---消息订阅与发布(pubsub)
  • 网络通信原理,进制转化总结
  • SQL Server数据库部署
  • 九、FreeRTOS之FreeRTOS列表和列表项
  • 基于ASP.NET MVC技术的图书管理系统的设计与实现
  • Discuz论坛自动采集发布软件
  • JS中 require 与 import 的区别
  • Android File Transfer for Mac:畅享强大的安卓文件传输工具
  • Sock0s1.1
  • ssh连接docker容器处理备忘
  • python处理日期和时间
  • C++模版