美文网首页
PHP的多进程--防止僵尸进程

PHP的多进程--防止僵尸进程

作者: weylau | 来源:发表于2019-08-29 15:09 被阅读0次

何为僵尸进程

僵尸进程是当子进程比父进程先结束,而父进程又没有回收子进程,释放子进程占用的资源,此时子进程将成为一个僵尸进程。如果父进程先退出 ,子进程被init接管,子进程退出后init会回收其占用的相关资源

僵尸进程带来的问题

僵尸进程由于父进程不回收系统保留的信息而一直占用着系统资源,其中有一项叫做进程描述符。系统通过分配它来启动一个进程。
但是系统所能使用的进程号是有限的,如果存在大量的僵尸进程,系统将因为没有可用的进程号而导致系统不能产生新的进程

如何查看僵尸进程

僵尸进程在系统中用 <defunct> 或 <z> 表示,通过 ps -ef 指令查看进程,如果发现某个进程的状态为 <defunct>/<z>,说明该进程是一个僵尸进程。

$ ps -ef | grep php
root      19355  17941  0 13:53 pts/3    00:00:00 php fork.php
root      19356  19355  0 13:53 pts/3    00:00:00 [php] <defunct>
root      19392  18316  0 13:53 pts/4    00:00:00 grep --color php

如何解决

  • 通过WAIT/WAITPID系统,等待子进程结束,并回收。

php代码实现

<?php
$pid = pcntl_fork();

if($pid == -1) {
    die('fork error');
}elseif($pid) {
    // 父进程阻塞着等待子进程的退出
    pcntl_wait($status);
    pcntl_waitpid($pid, $status);

    // 非阻塞方式,通过WNOHANG区分
    //pcntl_wait($status, WNOHANG);
    //pcntl_waitpid($pid, $status, WNOHANG);
    // 如没有上面的函数,在父进程sleep的时候,子进程就是僵尸进程
    sleep(20);
}else{
    cli_set_process_title("childTest");
    echo "child \r\n";
    exit;
}

/**
 *  int pcntl_wait ( int &$status [, int $options ] ):阻塞当前进程,直到当前进程的一个子进程退出或者收到一个结束当前进程的信号。
 *  int pcntl_waitpid ( int $pid , int &$status [, int $options ] ):功能同 pcntl_wait,区别为 waitpid 为等待指定 pid 的子进程。当 pid 为 -1
 * 时 pcntl_waitpid 与 pcntl_wait 一样。
 */
  • 通过SIGCHLD信号

当子进程结束后,系统会像父进程发送 SIGCHLD 信号给父进程,通知父进程子进程已经结束,但父进程默认不处理。我们可以在父进程收到这个信号时调用 wait()/waitpid() 来回收。

<?php
// 每执行一次低级语句会检查一次该进程是否有未处理过的信号
declare(ticks = 1);

// 信号处理函数
function sig_func() {
    echo "SIGCHLD \r\n";

    // 阻塞
    pcntl_wait($status);
    //pcntl_waitpid(-1, $status);

    // 非阻塞
    //pcntl_wait($status, WNOHANG);
    //pcntl_waitpid(-1, $status, WNOHANG);
}

pcntl_signal(SIGCHLD, 'sig_func');

$pid = pcntl_fork();

if($pid == -1) {
    die('fork error');
}elseif($pid) {
    // 父进程一直执行
    while(1) {
        sleep(5);
    }
}else{
    echo "child \r\n";
    exit;
}
  • 通过操作系统信号

如果父进程不关心子进程什么时候结束,那么可以用 pcntl_signal(SIGCHLD, SIG_IGN) 通知内核,自己对子进程的结束不感兴趣,那么子进程结束后,内核会回收,并不再给父进程发送信号。

<?php
// 每执行一次低级语句会检查一次该进程是否有未处理过的信号
declare(ticks = 1);

pcntl_signal(SIGCHLD, SIG_IGN);

$pid = pcntl_fork();

if($pid == -1) {
    die('fork error');
}elseif($pid) {
    // 父进程一直执行
    while(1) {
        sleep(5);
    }
}else{
    echo "child \r\n";
    exit;
}

参考文章:http://www.cnblogs.com/loveyouyou616/p/8881531.html

相关文章

  • PHP的多进程--防止僵尸进程

    何为僵尸进程 僵尸进程是当子进程比父进程先结束,而父进程又没有回收子进程,释放子进程占用的资源,此时子进程将成为一...

  • 僵尸进程处理方案

    僵尸进程介绍 Z(zombie)-僵尸进程(子进程终止,父进程没有wait子进程) 僵尸进程产生原因 僵尸进程是当...

  • 如何杀死僵尸进程?

    僵尸进程定义?查看僵尸进程,利用命令ps,可以看到有标记为Z(zombie)的进程就是僵尸进程。 僵尸进程的危害?...

  • 104-僵尸进程

    多进程编程要注意僵尸进程。子进程没有可执行代码后将变成僵尸进程,如果父进程一直运行,又没有处理僵尸进程的代码,僵尸...

  • 孤儿进程、僵尸进程与进程回收

    孤儿进程与僵尸进程 孤儿进程:父亲死了,子进程被init进程领养僵尸进程:子进程死了,父进程没有回收子进程的资源(...

  • linux上关闭僵尸进程

    查看所有僵尸进程: 之后kill -9杀死僵尸进程的父进程。

  • 进程之其他进程

    僵尸进程 定义 子进程先于父进程退出,父进程没有对子进程的退出做出相应的处理,此时子进程就会变成僵尸进程 影响 进...

  • 示例说明僵尸进程的危害及解决方法

    简述 首先简要说明下僵尸进程和孤儿进程的概念(前提都是父进程调用fork产生子进程) 僵尸进程:子进程终止,父进程...

  • 僵尸进程的产生和防范

    僵尸进程 僵尸进程的产生 fork产生的父进程和子进程有退出的先后顺序,如果子进程在父进程前退出就会产生,而父进程...

  • 如何清理僵尸进程(转载)

    首先说说:僵尸进程是什么? 僵尸进程是当子进程比父进程先结束,而父进程又没有回收子进程,释放子进程占用的资源,此时...

网友评论

      本文标题:PHP的多进程--防止僵尸进程

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