美文网首页
PHP信号处理的方式

PHP信号处理的方式

作者: 金星show | 来源:发表于2019-01-10 23:54 被阅读0次

这是pcntl模块初始化的代码。

PHP_MINIT_FUNCTION(pcntl)

{

   php_register_signal_constants(INIT_FUNC_ARGS_PASSTHRU);

   php_pcntl_register_errno_constants(INIT_FUNC_ARGS_PASSTHRU);

   php_add_tick_function(pcntl_signal_dispatch);

   return SUCCESS;

}

在这个函数的第三句话做了一件事,把pcntl_signal_dispatch这个函数注册成了tick的处理函数。而pcntl_signal_dispatch这个函数是做什么的呢?

void pcntl_signal_dispatch()

{

   ......

   /* Allocate */

   while (queue) {

           if (zend_hash_index_find(&PCNTL_G(php_signal_table), queue->signo, (void **) &handle)==SUCCESS) {

                   MAKE_STD_ZVAL(retval);

                   MAKE_STD_ZVAL(param);

                   ZVAL_NULL(retval);

                   ZVAL_LONG(param, queue->signo);

                   /* Call php signal handler - Note that we do not report errors, and we ignore the return value */

                   /* FIXME: this is probably broken when multiple signals are handled in this while loop (retval) */

                   call_user_function(EG(function_table), NULL, *handle, retval, 1, ¶m TSRMLS_CC);

                   zval_ptr_dtor(¶m);

                   zval_ptr_dtor(&retval);

           }

           next = queue->next;

           queue->next = PCNTL_G(spares);

           PCNTL_G(spares) = queue;

           queue = next;

   }

   ......

}

这个函数比较长,但核心的代码就是上面这个循环。php把注册的信号都放在一个queue里面,然后每次调用这个函数的时候,一个个来查看是否收到了信号需要处理,如果有信号的话,就调用相应的信号处理函数。

结合上面的初始化的代码,可以推测出php的信号处理函数是基于ticks来实现的,而不是注册到真正系统底层的信号处理函数中。而如果使用ticks的话,比如delare ticks=1, 那么每执行一条php语句都会调用上面的函数一次。而实际大部分时间里面并没有信号需要处理,所以这会造成极大的浪费。

那么,实际是这样吗?

首先,我们先写一个下面这样的代码。

image

然后,跑起来。我们可以通过kill命令给进程发送信号。通过kill -l得知SIGUSR1是30。然后,我们就向这个进程发送信号了。

image image

我发送了两次信号,程序打印了两次30。

说明这种方式是没问题的。

然后,我们注释掉delare那行。

再次跑起来,同样也是用kill来发送信号。

image image

这次啥也没收到了。

上面的这个实验证明了php的信号处理确实是基于ticks的。那么,各种php写的服务程序岂不是被这个拖累了?

确实,然而php > 4.3 版本以后,出现了pcntl_signal_dispatch()函数机制,即在主事件的循环里加入这个函数,就会自动检测信号并分发,效率相对较高。

事实上,一般需要信号处理的代码都是后端服务程序,而一般的后端服务程序都是按照事件处理的结构来编写的,也就是说,这种程序里面必定会有个主事件循环。在事件循环的每次循环中主动调用pcntl_signal_dispatch,就能基本实时的把信号处理掉,而且还能保证一个比较好的性能。

当然,你以为抛弃ticks,直接在事件循环中调用pcntl_signal_dispatch就是信号处理的正确打开方式吗?NONONO~,正确的编写需要php处理信号功能的代码,就应该去使用swoole,人家直接在c上面实现了信号处理,比pcntl不知道高到哪去了。
可以参考swoole_process模块。
https://wiki.swoole.com/wiki/page/p-process.html

相关文章

  • PHP信号处理的方式

    这是pcntl模块初始化的代码。 PHP_MINIT_FUNCTION(pcntl){ php_registe...

  • PHP信号处理

    PHP5.3.0起支持pcntl_signal系列进程控制函数可以对信号进行安装、分发等操作。 信号处理函数 pc...

  • Linux系统编程—信号捕捉

    前面我们学习了信号产生的几种方式,而对于信号的处理有如下几种方式: 默认处理方式; 忽略; 捕捉。 信号的捕捉,说...

  • 用PCNTL实现PHP多进程

    PHP的进程控制支持实现了Unix方式的进程创建, 程序执行, 信号处理以及进程的中断。 进程控制不能被应用在We...

  • 1-4节linux系统编程——进程通信信号和创建线程

    通信信号 信号:一种异步通信机制系统支持的信号都有默认的处理方式常用的信号及其处理:SIGINT | SIGQUI...

  • 10.1.2 自定义PHP的错误报告处理方式

    10.1.2 自定义PHP的错误报告处理方式 自定义错误报告的处理方式,可以完全绕过标准的PHP错误处理函数,这样...

  • linux信号以及core

    何为信号 信号(signal)用于通知进程发生了某种情况。进程有以下3种处理信号的方式: 忽略信号。有些信号表示硬...

  • OKhttp与php后台网络请求参数传数组

    最近公司后台换成php,从java到php其实没什么改变,只是个人的代码风格和思维方式不同,所以处理方式不同。其中...

  • 数字信号处理之卷积

    数字信号处理之卷积 1.数字信号处理原理 线性系统中,信号只能以乘以一个常数之后再相加的方式进行组合。例如,一个...

  • 你知道PHP信号处理的正确打开方式吗?

    今天翻PHP源码,无意中翻到了pcntl的源码,简单看了看,被吓了一跳。 这是pcntl模块初始化的代码。 PHP...

网友评论

      本文标题:PHP信号处理的方式

      本文链接:https://www.haomeiwen.com/subject/gxxtdqtx.html