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为什么选mysql作为数据库? Mysql 创建用户方法
Jul 02 PHP
php多任务程序实例解析
Jul 19 PHP
PHP批量查询WordPress留言者E-mail地址实现方法
Feb 15 PHP
php类常量用法实例分析
Jul 09 PHP
ucenter中词语过滤原理分析
Jul 13 PHP
浅谈PHP中try{}catch{}的使用方法
Dec 09 PHP
使用php自动备份数据库表的实现方法
Jul 28 PHP
微信公众号开发之获取位置信息php代码
Jun 13 PHP
php中目录操作opendir()、readdir()及scandir()用法示例
Jun 08 PHP
thinkPHP事务操作简单案例分析
Oct 17 PHP
laravel 框架结合关联查询 when()用法分析
Nov 22 PHP
php并发加锁问题分析与设计代码实例讲解
Feb 26 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使用file函数、fseek函数读取大文件效率对比分析
2016/11/04 PHP
javascript下判断一个元素是否存在的代码
2010/03/05 Javascript
说明你的Javascript技术很烂的五个原因
2011/04/26 Javascript
原生js操作checkbox用document.getElementById实现
2013/10/12 Javascript
深入领悟JavaScript中的面向对象
2013/11/18 Javascript
Javascript判断图片尺寸大小实例分析
2014/06/16 Javascript
JavaScript截取字符串的2个函数介绍
2014/08/27 Javascript
解决ueditor jquery javascript 取值问题
2014/12/30 Javascript
JavaScript实现搜索框的自动完成功能(一)
2016/02/25 Javascript
不能不知道的10个angularjs英文学习网站
2016/03/23 Javascript
jQuery树插件zTree使用方法详解
2017/05/02 jQuery
基于EasyUI的基础之上实现树形功能菜单
2017/06/28 Javascript
通过命令行创建vue项目的方法
2017/07/20 Javascript
Vue.js实现输入框绑定的实例代码
2017/08/24 Javascript
boostrap模态框二次弹出清空原有内容的方法
2018/08/10 Javascript
js实现随机点名程序
2020/09/17 Javascript
在VUE中实现文件下载并判断状态的方法
2019/11/08 Javascript
JS如何实现动态添加的元素绑定事件
2019/11/12 Javascript
Vue extend的基本用法(实例详解)
2019/12/09 Javascript
JS实现网页端猜数字小游戏
2020/03/06 Javascript
详解nginx配置vue h5 history去除#号
2020/11/09 Javascript
[01:06:43]完美世界DOTA2联赛PWL S3 PXG vs GXR 第二场 12.19
2020/12/24 DOTA
Python Unittest自动化单元测试框架详解
2018/04/04 Python
python实现简单的单变量线性回归方法
2018/11/08 Python
Django 响应数据response的返回源码详解
2019/08/06 Python
给大家整理了19个pythonic的编程习惯(小结)
2019/09/25 Python
python实现用类读取文件数据并计算矩形面积
2020/01/18 Python
html5 Canvas实现图片旋转的示例
2018/01/15 HTML / CSS
屈臣氏菲律宾官网:Watsons菲律宾
2020/06/30 全球购物
自我鉴定怎么写
2014/01/12 职场文书
六一儿童节活动总结
2014/08/27 职场文书
骨干教师申报材料
2014/12/17 职场文书
幼儿园保教工作总结2015
2015/10/15 职场文书
煤矿安全学习心得体会
2016/01/18 职场文书
使用Python拟合函数曲线
2022/04/14 Python
Rust中的Struct使用示例详解
2022/08/14 Javascript