Swoole的多进程模块

bVbvu4m?w=600&h=360

Swoole的多进程模块

介绍

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

Swoole\Process::__construct(callable $function, $redirect_stdin_stdout = false, $create_pipe = true)
  • $function,子进程建立成功后要执行的函数,底层会自动将函数保存到对象的callback属性上。若是但愿更改执行的函数,可赋值新的函数到对象的callback属性
  • $redirect_stdin_stdout,重定向子进程的标准输入和输出。启用此选项后,在子进程内输出内容将不是打印屏幕,而是写入到主进程管道。读取键盘输入将变为从管道中读取数据。默认为阻塞读取。
  • $create_pipe,是否建立管道,启用$redirect_stdin_stdout后,此选项将忽略用户参数,强制为true。若是子进程内没有进程间通讯,能够设置为 false。
swoole建立多进程很简单: new Swoole\Process('callback_function') 就能够了,好比我要同时建立6个进程,就for 循环6次就能够了。
假设前台给后台三组任务要求后台去执行,每一个任务大概须要执行一秒的时间,咱们利用多进程的形式去实现,让时间可以缩短。
for ($i = 0; $i < 6; $i++) {#建立了3个子进程
    $process = new Swoole\Process(function ($process) {
        sleep(1);
        echo PHP_EOL . posix_getpid() . PHP_EOL;#获取子进程PID
    }, false, true);
    $process->start();

进程间的通信

若是是很是简单的多进程执行任务,那么进程间就不须要通信了,实际状况下,不少业务是须要通信的,好比,发邮件,若是自进程发送失败了,那么是要通知主进程的等等。异步

swoole_process进程间支持2种通讯方式:函数

  1. 管道pipe
  2. 消息队列

管道通信

半双工: 数据单向流动, 一端只读, 一端只写。
同步 vs 异步: 默认为同步阻塞模式, 可使用 swoole_event_add() 添加管道到 swoole 的 event loop 中, 实现异步IOoop

clipboard.png

管道通讯是swoole_process默认的一种通讯方式。固然咱们也能够在实例化的时候经过参数来设定:学习

$process = new Swoole\Process('callback_function', false, true);

若是咱们打印$process会发现,每次建立一个进程后,就会随之建立一个管道,主进程想和哪个进程通讯,就向那个进程的管道写入/读取数据。spa

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

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

管道通信方式一:unix

$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());
}

管道通信方式二:code

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());//同步阻塞
}

消息队列的通信

消息队列:

  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);
  2. 投递数据到消息队列中:

    bool swoole_process->push(string $data);
  3. 从队列中提取数据

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

    案例:

    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;
    }

谢谢观赏

谢谢你们耐心观看,但愿对您有所帮助,也但愿你们提供下不一样的意见,找到更有效的方式来完成,共同窗习,谢谢!

相关文章
相关标签/搜索