php守护进程 加linux命令nohup实现任务每秒执行一次


Posted in PHP onJuly 04, 2011

Unix中 nohup 命令功能就是不挂断地运行命令,同时 nohup 把程序的所有输出到放到当前目录 nohup.out 文件中,如果文件不可写,则放到 <用户主目录>/nohup.out 文件中。那么有了这个命令以后我们php就写成shell 脚本使用循环来让我们脚本一直运行下去,不管我们终端窗口是否关闭都能够让我们php 脚本一直运行下去。
马上动手写个 PHP 小程序,功能为每30秒记录时间,写入到文件

# vi for_ever.php 
#! /usr/local/php/bin/php 
define('ROOT', dirname(__FILE__).'/'); 
set_time_limit(0); 
while (true) { 
file_put_contents(ROOT.'for_ever.txt', date('Y-m-d H:i:s')."\n", FILE_APPEND); 
echo date('Y-m-d H:i:s'), ' OK!'; 
sleep(30); 
} 
?>

保存退出,然后赋予 for_ever.php 文件可执行权限:
# chmod +x for_ever.php
让它在再后台执行:
# nohup /home/andy/for_ever.php.php &
记得最后加上 & 符号,这样才能够跑到后台去运行
执行上述命令后出现如下提示:
[1] 5157
nohup: appending output to 'nohup.out'
所有命令执行输出信息都会放到 nohup.out 文件中
这时你可以打开 for_ever.php 同目录下的 for_ever.txt 和 nohup.out 看看效果!
好了,它会永远运行下去了,怎么结束它呢?
# ps
PID TTY TIME CMD
4247 pts/1 00:00:00 bash
5157 pts/1 00:00:00 for_ever.php
5265 pts/1 00:00:00 ps
# kill -9 5157
找到进程号 5157 杀之,你将看到
[1]+ Killed nohup /home/andy/for_ever.php
OK!
====================
在很多项目中,或许有很多类似的后端脚本需要通过crontab定时执行。比如每10秒检查一下用户状态。脚本如下:
@file: /php_scripts/scan_userstatus.php
#!/usr/bin/env php -q 
$status = has_goaway(); 
if ($status) { 
//done 
} 
?>

通过crontab定时执行脚本scan_userstatus.php
#echo “*:*/10 * * * * /php_scripts/scan_userstatus.php”
这样,每隔10秒钟,就会执行该脚本。
我们发现,在短时间内,该脚本的内存资源还没有释放完,又启用了新的脚本。也就是说:新脚本启动了,旧脚本占用的资源还没有如愿释放。如此,日积月累,浪费了很多内存资源。我们对这个脚本进行了一下改进,改进后如下:
@file: /php_scripts/scan_userstatus.php
#/usr/bin/env php -q 
while (1) { 
$status = has_goaway(); 
if ($status) { 
//done 
} 
usleep(10000000); 
} 
?>

这样,不需要crontab了。可以通过以下命令执行脚本,达到相同的功能效果
#chmod +x /php_scripts/scan_userstatus.php
#nohup /php_scripts/scan_userstatus.php &
在这里,我们通过&将脚本放到后台运行,为了防止随着终端会话窗口关闭进程被杀,我们使用了nohup命令。那么有没有办法,不使nohup命令,也能够运行呢,就像Unin/Linux Daemon一样。接下来,就是我们要讲的守护进程函数。
什么是守护进程?一个守护进程通常补认为是一个不对终端进行控制的后台任务。它有三个很显著的特征:在后台运行,与启动他的进程脱离,无须控制终端。常用的实现方式是fork() -> setsid() -> fork() 详细如下:
@file: /php_scripts/scan_userstatus.php
#/usr/bin/env php -q 
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); 
} 
} 
?>

实现了守护进程函数以后,则可以建立一个常驻进程,所以只需要执行一次:
#/php_scripts/scan_userstatus.php
这里较为关键的二个php函数是pcntl_fork()和posix_setsid()。fork()一个进程,则表示创建了一个运行进程的副本,副本被认为是子进程,而原始进程被认为是父进程。当fork()运行之后,则可以脱离启动他的进程与终端控制等,也意味着父进程可以自由退出。 pcntl_fork()返回值,-1表示执行失败,0表示在子进程中,而返进程ID号,则表示在父进程中。在这里,退出父进程。setsid(),它首先使新进程成为一个新会话的“领导者”,最后使该进程不再控制终端,这也是成为守护进程最关键的一步,这意味着,不会随着终端关闭而强制退出进程。对于一个不会被中断的常驻进程来说,这是很关键的一步。进行最后一次fork(),这一步不是必须的,但通常都这么做,它最大的意义是防止获得控制终端。(在直接打开一个终端设备,而且没有使用O_NOCTTY标志的情况下, 会获得控制终端).
其它事项说明:
1) chdir() 将守护进程放到总是存在的目录中,另外一个好处是,你的常驻进程不会限制你umount一个文件系统。
2)umask() 设置文件模式,创建掩码到最大的允许限度。如果一个守护进程需要创建具有可读,可写权限的文件,一个被继承的具有更严格权限的掩码会有反作用。
3)fclose(STDIN), fclose(STDOUT), fclose(STDERR) 关闭标准I/O流。注意,如果有输出(echo),则守护进程会失败。所以通常将STDIN, STDOUT, STDERR重定向某个指定文件.
PHP 相关文章推荐
php下关于中英数字混排的字符串分割问题
Apr 06 PHP
php中计算中文字符串长度、截取中文字符串的函数代码
Aug 09 PHP
Yii操作数据库的3种方法
Mar 11 PHP
php防止sql注入之过滤分页参数实例
Nov 03 PHP
搭建Vim为自定义的PHP开发工具的一些技巧
Dec 11 PHP
PHP中的数组处理函数实例总结
Jan 09 PHP
Yii框架上传图片用法总结
Mar 28 PHP
php PDO异常处理详解
Nov 20 PHP
php技巧小结【推荐】
Jan 19 PHP
PHP使用第三方即时获取物流动态实例详解
Apr 27 PHP
win10 apache配置虚拟主机后localhost无法使用的解决方法
Jan 27 PHP
laradock环境docker-compose操作详解
Jul 29 PHP
ajax 的post方法实例(带循环)
Jul 04 #PHP
php高级编程-函数-郑阿奇
Jul 04 #PHP
php 日期和时间的处理-郑阿奇(续)
Jul 04 #PHP
php 目录与文件处理-郑阿奇(续)
Jul 04 #PHP
第4章 数据处理-php正则表达式-郑阿奇(续)
Jul 04 #PHP
第4章 数据处理-php字符串的处理-郑阿奇(续)
Jul 04 #PHP
第4章 数据处理-php数组的处理-郑阿奇
Jul 04 #PHP
You might like
PHP获得用户使用的代理服务器ip即真实ip
2006/12/31 PHP
php中socket的用法详解
2014/10/24 PHP
PHP获取QQ达人QQ信息的方法
2015/03/05 PHP
利用JS自动打开页面上链接的实现代码
2011/09/25 Javascript
JS 修改URL参数(实现代码)
2013/07/08 Javascript
jquery实现勾选复选框触发事件给input赋值
2015/02/01 Javascript
Javascript递归打印Document层次关系实例分析
2015/05/15 Javascript
nodejs 中模拟实现 emmiter 自定义事件
2016/02/22 NodeJs
ajax图片上传,图片异步上传,更新实例
2016/12/30 Javascript
vue.js选中动态绑定的radio的指定项
2017/06/02 Javascript
javascript checkbox/radio onchange不能兼容ie8处理办法
2017/06/13 Javascript
JavaScript学习笔记之函数记忆
2017/09/06 Javascript
基于vue开发的在线付费课程应用过程
2018/01/25 Javascript
Vue实现左右菜单联动实现代码
2018/08/12 Javascript
Vue利用Blob下载原生二进制数组文件
2019/09/25 Javascript
jQuery实现带进度条的轮播图
2020/09/13 jQuery
ant-design-vue 时间选择器赋值默认时间的操作
2020/10/27 Javascript
[01:13:59]LGD vs Mineski Supermajor 胜者组 BO3 第三场 6.5
2018/06/06 DOTA
Python实现删除Android工程中的冗余字符串
2015/01/19 Python
Python基于win32ui模块创建弹出式菜单示例
2018/05/09 Python
python调用tcpdump抓包过滤的方法
2018/07/18 Python
打包python 加icon 去掉cmd黑窗口方法
2019/06/24 Python
python数组循环处理方法
2019/08/26 Python
Python hashlib模块实例使用详解
2019/12/24 Python
python中68个内置函数的总结与介绍
2020/02/24 Python
Python求两个字符串最长公共子序列代码实例
2020/03/05 Python
解决import tensorflow as tf 出错的原因
2020/04/16 Python
PyCharm2020.1.2社区版安装,配置及使用教程详解(Windows)
2020/08/07 Python
存储过程的优点有哪些
2012/09/27 面试题
什么是反射?如何实现反射?
2016/07/25 面试题
资深地理教师自我评价
2013/09/21 职场文书
圣诞节开幕词
2015/01/29 职场文书
《学会生存》读后感3篇
2019/12/09 职场文书
新手必备Python开发环境搭建教程
2021/05/28 Python
Python Flask搭建yolov3目标检测系统详解流程
2021/11/07 Python
python如何利用cv2.rectangle()绘制矩形框
2022/12/24 Python