php 后台运行函数,php守护进程函数 后台执行脚本的实例详解
我們經常通過crontab定時執(zhí)行后端腳本。比如每10秒檢查一下用戶狀態(tài)。
例子:
復制代碼 代碼示例:
@file: /php_scripts/scan_userstatus.php
#!/usr/bin/env php -q? <?php ? $status = has_goaway();? if ($status) {????? //done? }? ?>
通過crontab定時執(zhí)行腳本scan_userstatus.php
復制代碼 代碼示例:
#echo “*:*/10 * * * * /php_scripts/scan_userstatus.php”
這樣,每隔10秒鐘,就會執(zhí)行該腳本。
在短時間內,該腳本的內存資源還沒有釋放完,又啟用了新的腳本。
即,新腳本啟動了,舊腳本占用的資源還沒有如愿釋放。
如此,日積月累,浪費了很多內存資源。
對這個腳本進行如下的改進:
復制代碼 代碼示例:
@file: /php_scripts/scan_userstatus.php
#/usr/bin/env php -q? <?php ? while (1) {????? $status = has_goaway();????? if ($status) {????????? //done????? }????? usleep(10000000);? }? ?>
這樣,不需要crontab了??梢酝ㄟ^以下命令執(zhí)行腳本,達到相同的功能效果
復制代碼 代碼示例:
#chmod +x /php_scripts/scan_userstatus.php
#nohup /php_scripts/scan_userstatus.php &
在這里,通過&將腳本放到后臺運行,為了防止隨著終端會話窗口關閉進程被殺,使用了nohup命令。
有辦法不使nohup命令,也能夠運行嗎,就像Unin/Linux Daemon一樣。
這里就要講到守護進程函數(shù)了。
什么是守護進程?一個守護進程通常補認為是一個不對終端進行控制的后臺任務。
它有三個很顯著的特征:在后臺運行,與啟動他的進程脫離,無須控制終端。
常用的實現(xiàn)方式是fork() -> setsid() -> fork() 詳細如下:
復制代碼 代碼示例:
@file: /php_scripts/scan_userstatus.php
#/usr/bin/env php -q? <?php ? daemonize();? while (1) {????? $status = has_goaway();????? if ($status) {????????? //done????? }????? usleep(10000000);? }???? function daemonize() {????? $pid = pcntl_fork();????? if ($pid === -1 ) {????????? return FALSE;????? } else if ($pid) {????????? usleep(500);????????? exit();??????????????? //exit parent????? }???????? chdir("/");????? umask(0);????? $sid = posix_setsid();????? if (!$sid) {????????? return FALSE;????? }???????? $pid = pcntl_fork();????? if ($pid === -1) {????????? return FALSE;????? } else if ($pid) {????????? usleep(500);????????? exit(0);????? }???????? if (defined('STDIN')) {????????? fclose(STDIN);????? }????? if (defined('STDOUT')){????????? fclose(STDOUT);????? }????? if (defined('STDERR')) {????????? fclose(STDERR);????? }? }? ?>
實現(xiàn)了守護進程函數(shù)以后,則可以建立一個常駐進程,所以只需要執(zhí)行一次:
復制代碼 代碼示例:
#/php_scripts/scan_userstatus.php
關鍵的二個php函數(shù)是pcntl_fork()和posix_setsid()。
fork()一個進程,則表示創(chuàng)建了一個運行進程的副本,副本被認為是子進程,而原始進程被認為是父進程。當fork()運行之后,則可以脫離啟動他的進程與終端控制等,也意味著父進程可以自由退出。
pcntl_fork()返回值,-1表示執(zhí)行失敗,0表示在子進程中,而返進程ID號,則表示在父進程中。在這里,退出父進程。setsid(),它首先使新進程成為一個新會話的“領導者”,最后使該進程不再控制終端,這也是成為守護進程最關鍵的一步,這意味著,不會隨著終端關閉而強制退出進程。對于一個不會被中斷的常駐進程來說,這是很關鍵的一步。進行最后一次fork(),這一步不是必須的,但通常都這么做,它最大的意義是防止獲得控制終端。(在直接打開一個終端設備,而且沒有使用O_NOCTTY標志的情況下, 會獲得控制終端).
總結:
1) chdir() 將守護進程放到總是存在的目錄中,另外一個好處是,你的常駐進程不會限制你umount一個文件系統(tǒng)。
2)umask() 設置文件模式,創(chuàng)建掩碼到最大的允許限度。
如果一個守護進程需要創(chuàng)建具有可讀,可寫權限的文件,一個被繼承的具有更嚴格權限的掩碼會有反作用。
3)fclose(STDIN), fclose(STDOUT), fclose(STDERR) 關閉標準I/O流。注意,如果有輸出(echo),則守護進程會失敗。
通常將STDIN, STDOUT, STDERR重定向某個指定文件.
創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎勵來咯,堅持創(chuàng)作打卡瓜分現(xiàn)金大獎總結
以上是生活随笔為你收集整理的php 后台运行函数,php守护进程函数 后台执行脚本的实例详解的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: php is_null(,PHP emp
- 下一篇: php 中 相关文章 的思路,WordP