PHP 使用pcntl和libevent 实现Timer功能_php实例_脚本之家

by admin on 2020年4月28日

完整更新列表:

  1. 要进行大量的网络耗时的操作2.
    要做大量的运算,并且,系统有多个cpu,为了让用户有更快的体验,把一个任务,分成几个小任务,最后合并。

PHP 扩展pcntl 实现 ” 多线程 ”pcntl 与 ticksticks 是通过 declare
{statement} 语法定义的 , declare 语法目前只能接受 ticks, 他定义的 ticks
= n 的意义是当 declare 指定的语句块中执行了 N 条低级语句去发生一个事件 ,
这个事件可以通过 register_tick_function 来注册 .pcntl 的信号机制是基于
ticks 机制实现的 . 因此 , 我们使用 pcntl 族函数中信号相关的函数时 ,
需要在前面增加 declare 语法结构 .int pcntl_alarm:$seconds
秒后向进程发送一个 SIGALRM 信号 , 每次调用 pcntl_alarm
方法都会取消之前设置的时钟 .void pcntl_exec(string $path[, array
$args[, array $env]]):在当前进程空间执行一个程序 .$path:
必须是二进制可执行文件 , 或具有有效脚本头信息 的脚本文件路径 .$args:
将要传递给该程序的字符串参数列表 $envs: 环境变量 . 以数组
方式传递给要执行程序的环境变量 .int pcntl_for k :创建一个子进程 ,
该子进程与父进程仅仅是 PID 不同 .在父线程执行时返回创建的子进程 pid,
在子线程执行时返回 0, 创建子进程失败时会在父进程上下文返回 -1, 并引发
php 错误 .理解这里的 fork 需要知道 : pcntl_fork 创建的是一个分支节点 ,
相当于一个标记 , 父进程完成后 , 子进程会从标记处继续执行 , 也就是说
pcntl_fork 后面的代码分别被父进程和子进程执行了两遍 ,
而两个进程在执行过程中得到的返回值是不同的 . 因此 ,
才可以分离父子进程执行不同的代码 .int pcntl_getpriority([int $pid =
getmypid()[, int $process_identifier = PRIO_PROCESS]]):获取给定
$pid 对应的进程的优先级 , 默认是通过 getmypid() 获取到的值也就是当前进程
.$pid: 如果没有指定 , 默认是当前进程 .$process_identifier: PRIO_PGRP,
PRIO_USER, PRIO_PROCESS 三者之一 , 默认 PRIO_PROCESS. 其中 PRIO_PGRP
指获取进程组的优先级 , PRIO_USER 指获取用户进程的优先级 , PRIO_PROCESS
指获取特定进程优先级 .返回进程的优先级 , 或者在发生错误时返回 false,
值越小说明越优先bool pcntl_setpriority(int $priority[, int $pid =
getmypid()[, int $process_identifier =
PRIO_PROCESS]]:设置进程的优先级 .$priority: 优先级值 , -20 到 20
的范围内 , 默认优先级为 0. 值越小说明越优先 .$pid: 如果没有指定 ,
指当前进程$process_identifier: 意义同 pcntl_getpriority 的
$process_identifier.设置成功返回 TRUE, 失败返回 FALSE.bool
pcntl_signal_dispatch:调用通过 pcntl_signal()
安装的即将发生的信号的处理器 .调用成功返回 TRUE, 失败返回 false.php
5.3.3 加入bool pcntl_signal(int $signo, callback $handler[, bool
$restart_syscalls = true]):为指定的信号 $signo 安装一个新的信号处理器
$handler.最后一个参数不明白意义 .bool pcntl_sigprocmask(int $how, array
$set[, array &$oldset]):增加 , 删除或设置锁信号 , 具体的行为依赖于
$how 参数$how: SIG_BLOCK 用于把信号增加到当前锁信号中 , SIG_UNBLOCK
用于把信号从当前锁信号中移除 , SIG_SETMASK
用于用给定的信号列表替换当前锁信号 .$set: 要增加 , 移除或设置的信号列表
.$oldset: 用于向调用者返回旧的锁定信号 .成功返回 TRUE, 失败返回
FALSE.int pcntl_sigtimedwait(array $set[, array &$siginfo[, int
$seconds = 0[, int $nanoseconds = 0]]]):pcntl_sigtimedwait 实际上和
pcntl_sigwaitinfo() 所做的是同样的事情 , 不过 pcntl_sigtimedwait
多了两个增强的参数 $seconds 和 $nanoseconds,
这样就允许脚本的停留时间有一个上限而不是无限制等待 .$set:
需要等待的信号列表$siginfo: 用来向调用者返回等待得到的信号的信息 ,
信息内容见 pcntl_sigwaitinfo$seconds: 超时的秒数$nanoseconds:
超时的纳秒数成功后 , pcntl_sigtimedwiat() 返回信号编号int
pcntl_sigwaitinfo(array $set[, array &$siginfo]):挂起当前脚本的执行 ,
直到接受到 $set 中的某个信号 , 如果其中的一个信号将要到达 ( 比如被
pcntl_sigprocmask 锁定 ) 那么 pcntl_sigwaitinfo 将会立刻返回$set:
等待的信号列表$siginfo: 用来向调用者返回等待得到的信号的信息 ,
该信息包含以下内容 :1. 所有信号都有以下三个信息 :a) signo: 信号编号b)
errno: 错误号c) code: 信号代码2. SIGCHLD 信号特有的信息a) status:
退出的值或信号b) utime: 用户消耗时间c) stime: 系统消耗时间d) pid:
发送进程 ide) uid: 发送进程的真实用户 id3. SIGILL, SIGFPE, SIGSEGV,
SIGBUS 拥有的信息a) addr: 产生故障的内存位置4. SIGPOLL 特有的信息 :a)
band: band event, 意义未知b) fd: 文件描述符函数成功运行返回信号编号int
pcntl_wait(int &$status[, int *options =
0]):挂起当前进程直到一个子进程退出或直到一个信号要求终止当前进程或调用一个信号处理函数
. 如果子进程在调用时已经退出 , 此函数会马上返回 ,
所有的系统资源都将被释放 .$status 用来保存子进程的状态信息 ,
该状态信息由以下函数产生 : pcntl_wifexited, pcntl_wifstopped,
pcntl_wifsignaled, pcntl_wexitstatus, pcntl_wtermsig,
pcntl_wstopsig.$options: 如果你的系统允许 wait3, 你可以提供一个可选的
options 参数 , 如果不提供这个参数 , wait 将会使用系统调用 ,
如果系统不允许 wait3, 提供这个参数不会有任何影响 , $options 的值可以是 0
或者 WNOHANG 和 WUNTRACED 两个常数 .函数返回退出的子进程的 PID,
或在错误时返回 -1, 或者如果提供 WNOHANG 作为 option 并且无有效子进程返回
0僵尸进程 : 由于父进程在 fork 之后 , 无法预知子进程什么时候结束 ,
所以子进程为了要留给父进程一些信息 , 会留下一个称作僵尸的数据结构 ,
等待由父进程发起 wait 的操作来为它收尸 , 在子进程结束
到父进程收尸前这一段时间子进程就被称为僵尸进程 , 在父进程结束后 ,
所有的子进程会交由 Init 来负责 , 因此 , 如果父进程结束 ,
僵尸进程都会被回收 , 但是 , 如果父进程永远不结束 ,
这些僵尸进程就一直占用进程号 , 如果系统进程号耗尽 ,
那么将导致无法启动新进程 , 因此 ,
安全的做法是在父进程中为自己产生的子进程去收尸 .int pcntl_waitpid(int
$pid, int &$status[, int $options = 0]):挂起当前进程直到给定 $pid
的子进程退出 , 或者当前进程接受到一个退出信号 , 或者接受到一 ige
信号去调用一个信号处理器 .如果给定 $pid
对应的子进程在调用此函数时已经退出 , 函数立刻返回 , 所有的系统资源被释放
.$pid: 进程号 , 小于 -1 表明等待的是进程组中的任何子进程 , 进程组号就是
$pid 的绝对值 . 等于 -1 表明等待任意紫禁城 , 与 pcntl_wait 函数行为一致
. 等于 0 代表等待与调用进程在同一组的子进程 , 大于 0 代表是特定的进程
.$status: 用来由函数返回子进程状态 . 该状态信息由以下函数产生 :
pcntl_wifexited, pcntl_wifstopped, pcntl_wifsignaled,
pcntl_wexitstatus, pcntl_wtermsig, pcntl_wstopsig.$options: 与
pcntl_wait 的 $options 意义相同int
pcntl_wexitstatus:返回一个已经中断的子进程返回代码 , 此函数仅在
pcntl_wifexited 函数返回 TRUE 时有用 .$status 参数是 pcntl_waitpid
产生的状态信息 .bool
pcntl_wifexited:检查给定状态是否表明子进程是正常退出的 .bool
pcntl_wifsignaled:检查给定状态是否表明子进程是由于收到某个信号退出的
.bool pcntl_wifstopped:检查 $status 是否能表明子进程当前已经停止 ,
这个函数只有在作用于 pcntl_waitpid 函数使用的 WUNTRACED 作为 $options
参数的值时产生的 $status 上才有效 .int pcntl_wstopsig:通过分析 $status
返回使得子进程停止的信号的编号 , 这个函数只有在 pcntl_wifsignaled 返回
TRUE 时才有效 .int pcntl_wtermsig:返回使进程中断的信号编号 .
这个函数只有在 pcntl_wifsignaled 返回 TRUE 时才有效 .

文章转载自:开源中国社区 []    

本文实例讲述了PHP多进程编程。分享给大家供大家参考,具体如下:

PHP 使用pcntl和libevent 实现Timer功能,先看例子,pcntl解释在下面。复制代码 代码如下:

  • Asynchronous Signal Handling (without ticks) in
    ext/pcntl.

  • Additional Context in pcntl_signal
    Handler

实例二:开多个子进程,避免fork泛滥注意:通过pcntl_XXX系列函数使用多进程功能。注意:pcntl_XXX只能运行在php CLI环境下,在web服务器环境下,会出现无法预期的结果,请慎用!更多关于PHP相关内容感兴趣的读者可查看本站专题:《PHP进程与线程操作技巧总结》、《PHP网络编程技巧总结》、《PHP基本语法入门教程》、《PHP数组操作技巧大全》、《php字符串用法总结》、《php+mysql数据库操作入门教程》及《php常见数据库操作技巧汇总》希望本文所述对大家PHP程序设计有所帮助。

PHP 7.1.0 Beta 1 发布了,此版本是7.1.0的第一个beta版。本次发布希望开发者能积极参与测试,并报告其错误和不兼容性。

实例一:

本次更新值得关注的内容:

pcntl_alarm — 为进程设置一个alarm闹钟信号pcntl_errno — 别名
pcntl_strerrorpcntl_exec — 在当前进程空间执行指定程序pcntl_fork —
创建子进程,在当前进程当前位置产生分支。译注:fork是创建了一个子进程,父进程和子进程
都从fork的位置开始向下继续执行,不同的是父进程执行过程中,得到的fork返回值为子进程
号,而子进程得到的是0。pcntl_get_last_error — Retrieve the error
number set by the last pcntl function which failedpcntl_getpriority —
获取任意进程的优先级pcntl_setpriority —
修改任意进程的优先级pcntl_signal_dispatch —
调用等待信号的处理器pcntl_signal — 安装一个信号处理器pcntl_sigprocmask
— 设置或检索阻塞信号pcntl_sigtimedwait —
带超时机制的信号等待pcntl_sigwaitinfo — 等待信号pcntl_strerror —
Retrieve the system error message associated with the given
errnopcntl_wait — 等待或返回fork的子进程状态pcntl_waitpid —
等待或返回fork的子进程状态pcntl_wexitstatus —
返回一个中断的子进程的返回代码pcntl_wifexited —
检查状态代码是否代表一个正常的退出。pcntl_wifsignaled —
检查子进程状态码是否代表由于某个信号而中断pcntl_wifstopped —
检查子进程当前是否已经停止pcntl_wstopsig —
返回导致子进程停止的信号pcntl_wtermsig — 返回导致子进程中断的信号

下载地址:

使用场景:

多进程常用函数:

第一步:

$ php -m 命令查看php是否安装pcntl 和 posix扩展,若没有则安装

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图