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

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

总结一下先~

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

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

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

上代码

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

$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);

}

}

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

59b21531277948.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)

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

正确代码之一

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就报错了。

更优处理办法

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信号。这些我还没接触。。。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
如果在使用`posix_spawn`函数创建解码进程时,发现新进程的环境变量为`NULL`,有可能是因为在创建新进程时,没有正确地设置环境变量。 在使用`posix_spawn`函数创建进程时,可以通过传递一个`posix_spawnattr_t`类型的对象来指定新进程的属性。这个对象有一个`envp`成员,它是一个字符指针数组,用于指定新进程的环境变量。如果在创建新进程时,这个成员被设置为`NULL`,那么新进程将没有任何环境变量。 如果想要继承父进程的环境变量,可以将父进程的`environ`数组作为参数传递给`posix_spawnattr_setenv`函数,然后将这个对象传递给`posix_spawn`函数。 以下是一个示例代码,用于在创建进程时正确地设置环境变量: ```c #include <spawn.h> #include <unistd.h> extern char **environ; int main(void) { posix_spawnattr_t attr; posix_spawn_file_actions_t fa; posix_spawnattr_init(&attr); posix_spawn_file_actions_init(&fa); // 将父进程的环境变量传递给新进程 posix_spawnattr_setflags(&attr, POSIX_SPAWN_SETPGROUP); posix_spawnattr_setenv(&attr, environ); // 设置其他属性,如文件描述符等 posix_spawn_file_actions_addopen(&fa, STDOUT_FILENO, "output.txt", O_WRONLY | O_CREAT | O_TRUNC, 0644); char *argv[] = { "/bin/ls", "-l", NULL }; pid_t pid; int ret = posix_spawn(&pid, "/bin/ls", &fa, &attr, argv, environ); if (ret != 0) { perror("posix_spawn"); return -1; } posix_spawnattr_destroy(&attr); posix_spawn_file_actions_destroy(&fa); return 0; } ``` 在这个示例代码中,通过将父进程的`environ`数组作为参数传递给`posix_spawnattr_setenv`函数,来设置新进程的环境变量。然后将这个`posix_spawnattr_t`类型的对象传递给`posix_spawn`函数,来创建一个新进程。新进程将会继承父进程的环境变量。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值