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调用数据库的存贮过程
Oct 09 PHP
《PHP编程最快明白》第四讲:日期、表单接收、session、cookie
Nov 01 PHP
php实现约瑟夫问题的方法小结
Mar 23 PHP
php转换颜色为其反色的方法
Apr 27 PHP
php数组索引与键值操作技巧实例分析
Jun 24 PHP
PHP中余数、取余的妙用
Jun 29 PHP
讲解WordPress中用于获取评论模板和搜索表单的PHP函数
Dec 28 PHP
PHP检测数据类型的几种方法(总结)
Mar 04 PHP
yii2 resetful 授权验证详解
May 18 PHP
php封装单文件上传到数据库(路径)
Oct 15 PHP
PHP通过GD库实现验证码功能示例
Feb 23 PHP
vmware linux系统安装最新的php7图解
Apr 14 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的网址
2006/11/25 PHP
关于在php.ini中添加extension=php_mysqli.dll指令的说明
2007/06/14 PHP
ajax实现无刷新分页(php)
2010/07/18 PHP
php使用fopen创建utf8编码文件的方法
2014/10/31 PHP
WordPress 插件——CoolCode使用方法与下载
2007/07/02 Javascript
认识延迟时间为0的setTimeout
2008/05/16 Javascript
js中switch case循环实例代码
2013/12/30 Javascript
javascript删除数组元素并且数组长度减小的简单实例
2014/02/14 Javascript
node.js中的require使用详解
2014/12/15 Javascript
jQuery使用元素属性attr赋值详解
2015/02/27 Javascript
JavaScript实现向右伸出的多级网页菜单效果
2015/08/25 Javascript
javascript伸缩型菜单实现代码
2015/11/16 Javascript
实例讲解js验证表单项是否为空的方法
2016/01/09 Javascript
JS+HTML5 canvas绘制验证码示例
2018/12/05 Javascript
vue中子组件传递数据给父组件的讲解
2019/01/27 Javascript
简单了解vue 插值表达式Mustache
2020/07/22 Javascript
如何利用javascript接收json信息并进行处理
2020/08/06 Javascript
python 输出一个两行字符的变量
2009/02/05 Python
python 统计列表中不同元素的数量方法
2018/06/29 Python
谈谈Python中的while循环语句
2019/03/10 Python
Python代码实现删除一个list里面重复元素的方法
2019/04/02 Python
Python实现去除图片中指定颜色的像素功能示例
2019/04/13 Python
如何利用Anaconda配置简单的Python环境
2019/06/24 Python
用Anaconda安装本地python包的方法及路径问题(图文)
2019/07/16 Python
详解用python生成随机数的几种方法
2019/08/04 Python
亚马逊中国官方网站:amazon.cn
2017/05/25 全球购物
党支部公开承诺践诺书
2014/03/28 职场文书
公开承诺书格式
2014/05/21 职场文书
保洁员岗位职责
2015/02/04 职场文书
公司年夜饭通知
2015/04/25 职场文书
劳动仲裁调解书
2015/05/20 职场文书
公司年会开场白
2015/06/01 职场文书
检讨书范文
2019/04/16 职场文书
js实现上传图片到服务器
2021/04/11 Javascript
关于k8s环境部署mysql主从的问题
2022/03/13 MySQL
springboot创建的web项目整合Quartz框架的项目实践
2022/06/21 Java/Android