記錄php daemon 進程 遇到的問題--posix_setsid函數

總結一下先~

  • 一個守護進程通常須要root權限,由於可能要使用特殊端口1-1024及其餘權限php

  • 一個守護進程的父進程會被fork以後被殺掉,因此能夠說他的父進程是init進程。html

  • 一個守護進程無需交互,也和終端(teriminalsession)無關,因此任何輸出,不管是向標準輸出仍是錯誤輸出,都須要特殊處理,涉及到的就是stdout和stderrlinux

上代碼

下面是我用php 編寫一個守護進程 demogit

<?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);
    }
}

好下面咱們執行操做,輸出以下ubuntu

root@tb:/home/tb/linuxing# php php_daemon.php 
fork succ
loop0
root@tb:/home/tb/linuxing# loop1
loop2
loop3
loop4
...
loop20

查看demo.txtsession

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 次數。。oop

爲何呢

進程從建立它的父進程那裏繼承了打開的文件描述符。如不關閉,將會浪費系統資源,(這卻是小事),形成進程所在的文件系統沒法卸下以及引發沒法預料的錯誤。
因此須要關閉這些.net

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

關閉標準輸入輸出與錯誤顯示。code

正確代碼之一

<?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);
    }


}

感謝

三金 CFC4N

守護進程詳解及建立,daemon()使用

不懂的太多

固然這只是個例子,實際中還須要考慮目錄權限,umask,figchld信號。這些我還沒接觸。。。

相關文章
相關標籤/搜索