本文為大家介紹了一下PHP守護進程化在C環境以及PHP環境下實現的方法以及實例代碼,一起來學習一下。
什么是守護進程?
一個守護進程通常補認為是一個不對終端進行控制的后臺任務。它有三個很顯著的特征:在后臺運行,與啟動他的進程脫離,無須控制終端。常用的實現方式是fork() -> setsid() -> fork()
在glibc里有一個函數daemon。調用此函數,就可使當前進程脫離終端變成一個守護進程,具體內容參見man daemon。PHP中暫時沒有此函數,PHP程序實現守護進程化有2種方法:
1.使用系統命令nohup
nohup php myprog.php > log.txt 這樣執行程序雖然也是轉為后臺運行,但實際上是依賴終端的,當用戶退出終端時進程就會被殺掉。需要使用nohup來實現
2.使用supervisor工具 (推薦此方案)
詳解supervisor使用教程
3.當然也可以用程序實現(不建議生產環境使用) C程序實現:
#include#include #include #include #include #include //實現守護進程步驟 void crete_daemon(void) { pid_t pid = 0; pid = fork(); if (pid<0) { perror("fork"); exit(-1); } if (pid > 0) { //1.父進程直接退出 exit(0); } //2. //執行到這里就是子進程 //setsid 將當前進程設置為一個新的會話期session,目的就是 //讓當前進程脫離控制臺,成為守護進程。 pid = setsid(); if (pid < 0) { perror("setsid"); exit(-1); } //3.設置當前進程的工作目錄為根目錄,不依賴于其他 chdir("/"); //4.umask設置為0確保將來進程有最大的文件操作權限 umask(0); //5.關閉文件描述符 //先要獲取當前系統中所允許打開的最大文件描述符數目 int i = 0; int cnt = sysconf(_SC_OPEN_MAX); for (i=0;i
測試結果:
守護進程:
這里較為關鍵的二個php函數是pcntl_fork()和posix_setsid()
fork()一個進程,則表示創建了一個運行進程的副本,副本被認為是子進程,而原始進程被認為是父進程。當fork()運行之后,則可以脫離啟動他的進程與終端控制等,也意味著父進程可以自由退出。 setsid(),它首先使新進程成為一個新會話的“領導者”,最后使該進程不再控制終端,這也是成為守護進程最關鍵的一步,這意味著,不會隨著終端關閉而強制退出進程。對于一個不會被中斷的常駐進程來說,這是很關鍵的一步。進行最后一次fork(),這一步不是必須的,但通常都這么做,它最大的意義是防止獲得控制終端。(在直接打開一個終端設備,而且沒有使用O_NOCTTY標志的情況下, 會獲得控制終端)
其它事項說明:
chdir() 守護進程默認繼承了父進程的當前工作目錄,當系統磁盤發生umount時將造成諸多的麻煩,通常將”/” 作為守護進程的當前工作目錄,可以避免上述的問題 umask() 守護進程默認繼承了父進程的文件權限掩碼,這就給該子進程使用文件帶來了諸多的麻煩。因此,把文件權限掩碼設置為0,可以大大增強該守護進程的靈活性 fclose(STDIN), fclose(STDOUT), fclose(STDERR) 關閉標準I/O流。用fork函數新建的子進程會從父進程那里繼承一些已經打開了的文件。這些被打開的文件可能永遠不會被守護進程讀寫,但它們一樣消耗系統資源,而且可能導致所在的文件系統無法卸下。