iis服务器助手广告
返回顶部
首页 > 资讯 > 精选 >Swoole中Process是什么
  • 325
分享到

Swoole中Process是什么

2023-06-06 15:06:58 325人浏览 八月长安
摘要

这篇文章主要介绍Swoole中Process是什么,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!本来计划开发 swoft 框架 中的 Process 模块, 所以需要对 swoole 的 Process 模块要有比较

这篇文章主要介绍Swoole中Process是什么,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!

本来计划开发 swoft 框架 中的 Process 模块, 所以需要对 swoole 的 Process 模块要有比较深入的了解才行. 不过根据 swoole 官方 wiki 的实践过程中, 一直有未理解的部分. 之前虽然也做过多次 多进程编程, 但是当真正需要进行框架开发的时候, 就会发现以前学到的知识不够全面, 无法指导整体的设计. 好在一直在坚持, 奉上现在理解的程度.

内容一览:

  • 进程相关基础操作: fork/exit/kill/wait

  • 进程相关高级操作: 主进程退出子进程干完活后也退出; 子进程异常退出主进程自动重启

  • 进程间通信(IPC) - 管道(pipe)

  • 进程间通信(IPC) - 消息队列(message queue)

  • swoole process 模块提供的更多功能

进程相关基础操作

进程是什么: 进程是运行者的程序

先来看看一个最简单的例子:

<?PHPecho posix_getpid(); // 获取当前进程的 pidswoole_set_process_name('swoole process master'); // 修改所在进程的进程名sleep(100); // 模拟一个持续运行 100s 的程序, 这样就可以在进程中查看到它, 而不是运行完了就结束

通过 ps aux 查看进程:

未设置进程名

设置进程名

再来看看 swoole 中使用子进程的基础操作:

use Swoole\Process;$process = new Process(function (Process $worker) {    if (Process::kill($worker->pid, 0)) { // kill操作常用来杀死进程, 传入 0 可以用来检测进程是否存在        $worker->exit(); // 退出子进程    }});$process->start(); // 启动子进程Process::wait(); // 回收退出的子进程
  • new Process(): 通过回调函数来设置子进程将要执行的逻辑

  • $process->start(): 调用 fork() 系统调用, 来生成子进程

  • Process::kill(): kill操作给进程发送信号, 常用来杀死进程, 传入 0 可以用来检测进程是否存在

  • Process::wait(): 调用 wait() 系统调用, 回收子进程, 如果不回收, 子进程会编程 僵尸进程, 浪费系统资源

  • $worker->exit(): 子进程主动退出

我在这里有一个疑问:

主进程的生命周期是怎么样的? 子进程的生命周期是怎么样的?

有这样一个疑问也来自于我之前的思维惯性: 理解一个事物时从事物的生命周期进行理解. 结合 进程是运行着的程序 来一起理解:

  • new Process(): 只有回调函数的逻辑会在进程中执行

  • 除此之外的代码都是在主进程中执行

进程相关高级操作

  • 主进程退出子进程干完活后也退出

  • 子进程异常退出主进程自动重启

<?phpuse Swoole\Process;class MyProcess1{    public $mpid = 0; // master pid, 即当前程序的进程ID    public $works = []; // 记录子进程的 pid    public $maxProcessNum = 1;    public $newIndex = 0;    public function __construct()    {        try {            swoole_set_process_name(__CLASS__. ' : master');            $this->mpid = posix_getpid();            $this->run();            $this->processWait();        } catch (\Exception $e) {            die('Error: '. $e->getMessage());        }    }    public function run()    {        for ($i=0; $i<$this->maxProcessNum; $i++) {            $this->createProcess();        }    }    public function createProcess($index = null)    {        if (is_null($index)) {            $index = $this->newIndex;            $this->newIndex++;        }        $process = new Process(function (Process $worker) use($index) { // 子进程创建后需要执行的函数            swoole_set_process_name(__CLASS__. ": worker $index");            for ($j=0; $j<3; $j++) { // 模拟子进程执行耗时任务                $this->checkMpid($worker);                echo "msg: {$j}\n";                sleep(1);            }        }, false, false); // 不重定向输入输出; 不使用管道        $pid = $process->start();        $this->works[$index] = $pid;        return $pid;    }    // 主进程异常退出, 子进程工作完后退出    public function checkMpid(Process $worker) // demo中使用的引用, 引用表示传的参数可以被改变, 由于传入 $worker 是 \Swoole\Process 对象, 所以不用使用 &    {        if (!Process::kill($this->mpid, 0)) { // 0 可以用来检测进程是否存在            $worker->exit();            $msg = "master process exited, worker {$worker->pid} also quit\n"; // 需要写入到日志中            file_put_contents('process.log', $msg, FILE_APPEND); // todo: 这句话没有执行        }    }    // 重启子进程    public function rebootProcess($pid)    {        $index = array_search($pid, $this->works);        if ($index !== false) {            $newPid = $this->createProcess($index);            echo "rebootProcess: {$index}={$pid}->{$newPid} Done\n";            return;        }        throw new \Exception("rebootProcess error: no pid {$pid}");    }    // 自动重启子进程    public function processWait()    {        while (1) {            if (count($this->works)) {                $ret = Process::wait(); // 子进程退出                if ($ret) {                    $this->rebootProcess($ret['pid']);                }            } else {                break;            }        }    }}new MyProcess1();

说明以下几点:

  • 子进程运行结束后就会退出, 通过 Process::wait() 检测到子进程退出信号执行自动重启, 子进程就会一直执行下去

  • 关于函数参数传 引用/指针, 一个很好的理解方式是: 参数可以被修改

运行并模拟主进程异常退出:

模拟主进程异常退出

输出

进程间通信(IPC) - 管道(pipe)

管道的几个关键词:

  • 半双工: 数据单向流动, 一端只读, 一端只写.

  • 同步 vs 异步: 默认为同步阻塞模式, 可以使用 swoole_event_add() 添加管道到 swoole 的 event loop 中, 实现异步IO

  • 管道类型(数据格式): SOCK_STREAM, 流式, 需要用户自己处理数据的封包/解包; SOCK_DGRAM, 数据报, 每次收发都是一次完整的数据包 (DGRAM/STREAM)

注意, swoole wiki - process->write() 中提到 SOCK_DGRAM 并不会乱序丢包

先来看一个简单的例子, php从shell管道中读取数据:

// get pip data$fp = fopen('php://stdin', 'r');if ($fp) {    while ($line = fgets($fp, 4096)) {        echo "php get pip data: ". $line;    }    fclose($fp);}

从shell管道读取数据

swoole process中的管道很强大, 支持 子进程写, 主进程读 以及 主进程写, 子进程读:

use Swoole\Process;// 子进程写, 父进程读$process = new Process(function (Process $worker) {    $worker->write("worker");});$process->start();$msg = $process->read();echo "from process: $msg", "\n";// 父进程写, 子进程读$process = new Process(function (Process $worker) {    $msg = $worker->read();    echo "from master: $msg", "\n";});$process->start();$process->write('master');

使用管道多次读写

注意区分 $worker->write()$process->write(), 之前一直错误的以为这 2 个是相同的, 其实就是把 $process 误以为是子进程, 从而相当于 $process->write() 就是子进程写管道 -- 其实这里是主进程内执行的逻辑, 是主进程写数据到管道, 供子进程读取

swoole中其他管道相关操作:

  • 异步IO

use Swoole\Process;use Swoole\Event;// 异步IO$process = new Process(function (Process $worker) {    $GLOBALS['worker'] = $worker;    Event::add($worker->pipe, function (int $pipe) { // 使用 swoole_event_add 添加管道到异步IO                $worker = $GLOBALS['worker'];        $msg = $worker->read();        echo "from master: $msg \n";        $worker->write("hello master");        sleep(2);        $worker->exit(0);    });});$process->start();$process->write("master msg 1");$msg = $process->read();echo "from process: $msg \n";

异步IO

  • 设置超时

use Swoole\Process;// 设置管道超时$process = new Process(function (Process $worker) {    sleep(5);});$process->start();$process->setTimeout(0.5);$ret = $process->read();var_dump($ret);var_dump(swoole_errno());

管道超时

插播一个趣事, @thinkpc 看完 2017北京PHP开发者年会, 就知道为啥会点赞了

  • 关闭管道

// 关闭管道: 默认值0->关闭读写 1->关闭写 2->关闭读$process->close();

进程间通信(IPC) - 消息队列(message queue)

消息队列:

  • 一系列保存在内核中的消息链表

  • 有一个 msgKey, 可以通过此访问不同的消息队列

  • 有数据大小限制, 默认 8192, 可以通过内核修改

  • 阻塞 vs 非阻塞: 阻塞模式下 pop()空消息队列/push()满消息队列会阻塞, 非阻塞模式可以直接返回

swoole 中使用消息队列:

  • 通信模式: 默认为争抢模式, 无法将消息投递给指定子进程

  • 新建消息队列后, 主进程就可以使用

  • 消息队列不可和管道一起使用, 也无法使用 swoole event loop

  • 主进程中要调用 wait(), 否则子进程中调用 pop()/push() 会报错

use Swoole\Process;$process = new Process(function (Process $worker) {    // $worker->push('worker');    echo "from master: ". $worker->pop(). "\n";    sleep(2);    // $worker->exit();}, false, false); // 关闭管道// 参数一为 msgKey, 这里是默认值// 参数二为 通信模式, 默认值 2 表示争抢模式, 这里还加上了 非阻塞$process->useQueue(ftok(__FILE__, 1), 2| Process::IPC_NOWAIT);$process->push('hello1'); // 使用 useQueue 后, 主进程就可以读写消息队列了$process->push('hello2');echo "from woker: ". $process->pop(). "\n";// echo "from woker: ". $process->pop(). "\n";$process->start(); // 启动子进程// 消息队列状态var_dump($process->statQueue());// 删除队列, 如果不调用则不会在程序结束时清楚数据, 下次使用相同 msgKey 时还可以访问数据$process->freeQueue();var_dump(Process::wait()); // 要调用 wait(), 否则子进程中 push()/pop() 会报错

消息队列

swoole process 模块提供的更多功能

  • swoole_set_process_name(): 修改进程名, 不兼容 Mac

  • swoole_process->exec(string $execfile, array $args) 执行外部程序

参数 $execfile 需要使用可执行文件的绝对路径, 参数 args 为参数数组

// 比如 python test.py 123swoole_process->exec('/usr/bin/Python', ['test.py', 123]);// 更复杂的例子swoole_process->exec(('/usr/local/bin/php', ['/var/www/project/yii-best-practice/cli/yii', 't/index', '-m=123', 'abc', 'xyz']);// 父进程 exec 进程进行管道通信use Swoole\Process;$process = new Process(function (Process $worker) {    $worker->exec('/bin/echo', ['hello']);    $worker->write('hello');}, true); // 需要启用标准输入输出重定向$process->start();echo "from exec: ". $process->read(). "\n";

父进程与exec进程通过管道通信

  • \Swoole\Process::kill($pid, $signo = SIGTERM): 向指定进程发送信号, 默认是终止进程, 传 0 可检测进程是否存在

  • \Swoole\Process::wait(): 回收子进程, 如果主进程不调用此方法, 子进程会变成 僵尸进程, 浪费系统资源

  • \Swoole\Process::signal(): 异步信号监听

use Swoole\Process;// 异步信号监听 + waitProcess::signal(SIGCHLD, function ($signal) { // 监听子进程退出信号    // 可能同时有多个子进程退出, 所以要while循环    while ($ret = Process::wait(false)) { // false 表示不阻塞        var_dump($ret);    }});

\Swoole\Process::daemon(): 将当前进程变为一个守护进程

use Swoole\Process;// daemonProcess::daemon();swoole_set_process_name('test daemon process');sleep(100);

daemon-守护进程

  • \Swoole\Process::alarm(): 高精度定时器(微秒级), 对 setitimer 系统调用的封装, 可以配合 \Swoole\Process::signal() / pcntl_signal 使用

注意不可和 \Swoole\Timer 同时使用

// signal + alarm// 第一个参数表示时间, 单位 us, -1 表示清除定时器// 第二个参数表示类型 0->真实时间->SIGALAM 1->cpu时间->SIGVTALAM 2->用户态+内核态时间->SIGPROFProcess::alarm(100*1000); // 100msProcess::signal(SIGALRM, function ($signal) {    static $i = 0;    echo "#$i \t alarm \n";    $i++;    if ($i>20) {        Process::alarm(-1); // -1 表示清除    }});

alarm

  • \Swoole\Process::setaffinity(): 设置CPU亲和, 即将进程绑定到指定CPU核上

传值范围: [0, swoole_cpu_num())
CPU亲和: CPU的速度远远高于IO的速度, 所以CPU有多级缓存来解决IO等待的问题, 绑定指定CPU, 更容易命中CPU缓存

写在最后

资源推荐:

  • 图灵社区 - 理解UNIX进程 + 「理解Unix进程」读书笔记

  • blog - 「进程」编程

todo:

  • 使用输入输出重定向

  • 管道类型为 SOCK_STREAM 时的情况, 是否需要 封包/解包 处理, 即 swoole wiki - process->write() 中提到的 管道通信默认的方式是流式,write写入的数据在read可能会被底层合并

  • 多进程 + 异步IO 的注意事项

能理解 因为子进程会继承父进程的内存和IO句柄 这个会产生的影响, 但是给的示例并没有说明这个问题

use Swoole\Process;use Swoole\Event;// 多个子进程 + 异步IO$workers = [];$workerNum = 3;for ($i=0; $i<$workerNum; $i++) {    $process = new Process(function (Process $worker) {        $worker->write($worker->pid);        echo "worker: {$worker->pid} \n";    });    $pid = $process->start();    $workers[$pid] = $process;    // Event::add($process->pipe, function (int $pipe) use ($process) {    //  $data = $process->read();    //  echo "recv: $data \n";    // });}foreach ($workers as $worker) {    Event::add($worker->pipe, function (int $pipe) use ($worker) {        $data = $worker->read();        echo "recv: $data \n";    });}

多进程异步IO

以上是“Swoole中Process是什么”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注编程网精选频道!

--结束END--

本文标题: Swoole中Process是什么

本文链接: https://www.lsjlt.com/news/247574.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Swoole中Process是什么
    这篇文章主要介绍Swoole中Process是什么,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!本来计划开发 swoft 框架 中的 Process 模块, 所以需要对 swoole 的 Process 模块要有比较...
    99+
    2023-06-06
  • oracle中的v$process是什么
    这篇文章主要介绍“oracle中的v$process是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“oracle中的v$process是什么”文章能帮助大家解决...
    99+
    2024-04-02
  • php中Swoole的模块是什么
    这篇文章给大家分享的是有关php中Swoole的模块是什么的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。1、Server强大的TCP/UDP Server框架,支持多线程,EventLoop,事件驱动,异步,Wo...
    99+
    2023-06-15
  • 什么是System Idle Process
    什么是System Idle Process,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。问:我发现Windows任务管理器中有用户名为System的文件在运行...
    99+
    2023-06-14
  • python中process函数的用法是什么
    在Python中,process函数是multiprocessing模块中的一个函数,主要用于创建和管理子进程。process函数的...
    99+
    2023-10-20
    python
  • nodejs中process进程的作用是什么
    这期内容当中小编将会给大家带来有关nodejs中process进程的作用是什么,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。  进程模块  process模块允许你获得或...
    99+
    2024-04-02
  • swoole协程的概念是什么
    本篇内容介绍了“swoole协程的概念是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!swoole协程是更加轻量级的线程,一个线程可以拥...
    99+
    2023-06-29
  • swoole能干什么
    这篇文章主要介绍“swoole能干什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“swoole能干什么”文章能帮助大家解决问题。swoole是面向生产环境的PHP异步网络通信引擎,是为了提升网站的...
    99+
    2023-06-29
  • swoole是什么及有哪些功能
    这篇文章主要介绍“swoole是什么及有哪些功能”,在日常操作中,相信很多人在swoole是什么及有哪些功能问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”swoole是什么及有哪些功能”的疑惑有所帮助!接下来...
    99+
    2023-07-04
  • swoole能做什么
    这篇文章主要讲解了“swoole能做什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“swoole能做什么”吧!swoole的好处:1、多个Web工作器和单独的任务工作器,从而可以延迟代码;...
    99+
    2023-06-29
  • swoole协程的适用场景是什么
    这篇文章主要介绍“swoole协程的适用场景是什么”,在日常操作中,相信很多人在swoole协程的适用场景是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”swoole协程的适用场景是什么”的疑惑有所帮助!...
    99+
    2023-06-29
  • swoole协程的调度方式是什么
    这篇文章主要讲解了“swoole协程的调度方式是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“swoole协程的调度方式是什么”吧!在swoole中,因为协程的切换是串行的,在同一个时间...
    99+
    2023-06-29
  • swoole不能用sleep的原因是什么
    这篇文章主要介绍“swoole不能用sleep的原因是什么”,在日常操作中,相信很多人在swoole不能用sleep的原因是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”swoole不能用sleep的原因...
    99+
    2023-06-30
  • swoole协程的执行流程是什么
    今天小编给大家分享一下swoole协程的执行流程是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。在swoole中,Swo...
    99+
    2023-06-29
  • 在Swoole上运行Laravel的原因是什么
    这篇文章主要介绍“在Swoole上运行Laravel的原因是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“在Swoole上运行Laravel的原因是什么”文章能帮助大家解决问题。Swoole 是...
    99+
    2023-06-30
  • php要用swoole扩展的原因是什么
    这篇文章主要介绍“php要用swoole扩展的原因是什么”,在日常操作中,相信很多人在php要用swoole扩展的原因是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”php要用swoole扩展的原因是什么...
    99+
    2023-07-05
  • web服务器不用swoole的原因是什么
    这篇文章主要介绍“web服务器不用swoole的原因是什么”,在日常操作中,相信很多人在web服务器不用swoole的原因是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”web服务器不用swoole的原因...
    99+
    2023-07-05
  • swoole和redis有什么区别
    本篇内容主要讲解“swoole和redis有什么区别”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“swoole和redis有什么区别”吧!区别:1、redis是由“Salvatore Sanfi...
    99+
    2023-06-29
  • 在Oracle中session和process的区别什么
    这篇文章将为大家详细讲解有关在Oracle中session和process的区别什么,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。对应元数据表v$resou...
    99+
    2024-04-02
  • swoole框架有什么功能
    这篇文章主要讲解了“swoole框架有什么功能”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“swoole框架有什么功能”吧!swoole是一种PHP协程框架,是一个面向生产环境的PHP异步网...
    99+
    2023-06-28
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作