PHP程序员学习笔记|如何学习PHP

个人总结的PHP学习方法


伟大的PHP-进程守护 posix_setsid函数

2018-7-11 0phpcom 程序人生


总结一下先~

  • 一个守护进程一般需要root权限,因为可能要使用特殊端口1-1024及其他权限

  • 一个守护进程的父进程会被fork之后被杀掉,所以可以说他的父进程是init进程。

  • 一个守护进程无需交互,也和终端(teriminalsession)无关,所以任何输出,无论是向标准输出还是错误输出,都需要特殊处理,涉及到的就是stdout和stderr



上代码

下面是我用php 编写一个守护进程 demo

<?php

$pid = pcntl_fork();
if ($pid < 0) {
    echo 'fork error';
    exit;
} else if ($pid) {
    //父进程执行到此,返回的为子进程pid,此时需要把父进程杀掉。
    echo "fork succ\n";
    exit;
} else {// 重点是posix_setsid()
/*
 http://linux.die.net/man/2/setsid
 [setsid详解][1] 主要目的脱离终端控制,自立门户。
创建一个新的会话,而且让这个pid统治这个会话,他既是会话组长,也是进程组长。
而且谁也没法控制这个会话,除了这个pid。当然关机除外。。
这时可以成做pid为这个无终端的会话组长。
注意这个函数需要当前进程不是父进程,或者说不是会话组长。
在这里当然不是,因为父进程已经被kill
*/     
    $sid = posix_setsid();
    if ($sid < 0) {
        echo 'setsid error';
        exit;
    }
    //下面是进行的的daemon test
    for ($i = 0; $i <= 20; $i++) {
        echo 'loop' . $i . "\n";
        file_put_contents('demo.txt', 
        $i . "--" . date("Y-m-d H:i:s", time()) . "\n", FILE_APPEND);
        sleep(1);
    }
}

好下面我们执行操作,输出如下
11.png

查看demo.txt

cat demo.txt 
0--2016-07-15 17:49:47
1--2016-07-15 17:49:48
2--2016-07-15 17:49:49
...

问题来了

貌似没有问题,但是用以上php代码,执行后,马上关闭当前终端。则发现程序并不会完整输出20行数据,只是部分数据。
问题复现步骤:
1.ubuntu终端Azhong 执行 php php_daemon.php
2.关闭终端A
3.打开新终端B,ps -aux |grep php 发现无此进程

如果手慢,自己把握时间或者调整for 次数。。


为什么呢

进程从创建它的父进程那里继承了打开的文件描述符。如不关闭,将会浪费系统资源,(这倒是小事),造成进程所在的文件系统无法卸下以及引起无法预料的错误。
所以需要关闭这些

fclose(STDIN),fclose(STDOUT),fclose(STDERR)

关闭标准输入输出与错误显示。

正确代码之一

<?php $pid = pcntl_fork(); if ($pid < 0) {
    echo 'fork error'; exit;
} else if ($pid) {
    echo "fork succ\n"; exit;

} else { $sid = posix_setsid(); if ($sid < 0) {
        echo 'setsid error'; exit;
    } for ($i = 0; $i <= 20; $i++) { // echo 'loop' . $i . "\n";
        file_put_contents('demo.txt', $i . "--" . date("Y-m-d H:i:s", time()) . "\n", FILE_APPEND);
        sleep(1);
    }

}

再解释一下

如果想在关闭当前终端后继续执行
需要关闭echo 那一行,因为当然echo 和当然session关联,sesssion关闭后,echo就会导致php致命错误,所以下面的file_put_contents不会执行。

所以为了避免除显示输出的echo导致php错误的问题,我们一般建议这样

 global $STDOUT, $STDERR; fclose(STDOUT); fclose(STDERR);
      $STDOUT = fopen('/dev/null', "rw+");
      $STDERR = fopen('/dev/null', "rw+"); 

加上上面那句,所有的显示的不显示的echo err之类都可以被忽略。也就是说你把
echo 'loop' . $i . "n";这句加上也没有问题
指到dev/null,,如果你不这样,你的stdout会跟你的session有关。。
你的session一关,你的stdout就失效,,echo就报错了。

更优处理办法

<?php $pid = pcntl_fork(); if ($pid < 0) {
    echo 'fork error'; exit;
} else if ($pid) {
    echo "fork succ\n"; exit;

} else { $sid = posix_setsid(); if ($sid < 0) {
        echo 'setsid error'; exit;
    }
      global $STDOUT, $STDERR;
          fclose(STDOUT);
          fclose(STDERR); $STDOUT = fopen('/dev/null', "rw+"); $STDERR = fopen('/dev/null', "rw+"); for ($i = 0; $i <= 20; $i++) {
        file_put_contents('demo.txt', $i . "--" . date("Y-m-d H:i:s", time()) . "\n", FILE_APPEND);
        sleep(1);
    }


} 

不懂的太多

当然这只是个例子,实际中还需要考虑目录权限,umask,figchld信号。这些我还没接触。。。

« [转]一个老程序员PHP程序员说的话 | 【php】session之阻塞»
发表评论:


站点统计
  • 建站日期: 2015-2-10 18:22
  • 运行时间: 3324 天
  • 日志总数: 221 篇
  • 置顶日志: 0 篇
  • 草稿数目: 15 篇
  • 待审文章: 0 篇
  • 页面数量: 0 个
  • 评论数量: 53 条
  • 微语数量: 5 条
  • 微语评论: 0 条
  • 友链数量: 1 个
  • 加密文章: 1 篇
  • 分类总量: 10 个
  • 父分类数: 10 个
  • 子分类数: 0 个
  • 用户数量: 1 人
  • 管理员数: 1 人
  • 驻站作者: 0 位
  • 标签数量: 11 个
  • 附件总量: 223 件
  • 最后发表: 2021-5-8 16:17



订阅Rss