linux/unix下使用php来做守护进程

网络整理 - 08-17

[  起源 ]

Linux/Unix下守护进程(Daemon)大家都知道,比如我们常用的httpd、mysqld等等,就是常驻内存运行的程序,类似于Windows下的服务。一般守护进程都是使用C/C++来写,就是通过fork生成子进程,当前台shell下的父进程被杀掉,子进程就转到后台运行,为了不在终端产生输出信息,就通过syslog等函数来写日志文件。

我们知道php是脚本语言,通过php的脚本引擎来执行,所以要做成守护进程比较麻烦,我们今天就来结合Unix/Linux的命令来实现我们守护进程的功能。

 [  原理 ]

Unix中的nohup命令的功能就是不挂断地运行命令,同时nohup把程序的所有输出到放到当前目录的nohup.out文件中,如果文件不可写,则放到<用户主目录>/nohup.out 文件中。那么有了这个命令以后,我们的php程序就写程shell脚本,使用循环来让我们的脚本一直运行,那么不管我们终端窗口是否关闭,都能够让我们的php脚本一直运行。当然,当我们的php进程被杀或者我们的操作系统重启了,自然就会中止了。

 [  功能 ]

肯定会问,让我们的php脚本做了守护进程又有什么用处呢?当然有,比如最典型的作用,能够基本的替代cron的功能,比如我们需要定期实行的某些操作,完全可以交给它来做,不再需要cron,当然,如果服务器重启就没有办法了,不过,一般的Unix服务器不是那么容易重启的。另外,我们还可以做一个简单的服务器端的功能,比如做一个能够Telnet过去的服务器,嘿嘿,可以做成一个小后门,不过这样实现稍微有点复

杂。

[  实践 ]

例子:自动生成文件

我们现在来做两个例子来证明我们上面的说法。首先第一个是每个三十秒自动生成一个文件,永远执行下去。

首必须确保操作系统是Unix或者Linux,比如可以是FreeBSD、Redhat、Fedora或者SUSE什么的。然后我们必须确保我们的php脚本引擎是在 /usr/local/php/bin/php,具体路径可以按照你实际路径来写,如果没有脚本引擎,请自行安装。

比如当前目录是 /home/heiyeluren/,那么我们使用vi或者其他编辑器编写一个叫做php_daemon1.php的文件:
$ vi php_daemon1.php

然后写入如下代码:
#! /usr/local/php/bin/php
<?
set_time_limit(0);
while(1)
{
 @fopen("test_".time().".txt","w");
 sleep(30);
}
?>

然后保存并且退出vi,然后赋予php_daemon1.php文件可执行权限:
$ chmod +x /home/heiyeluren/php_daemon1.php

然后再让我们的脚本再后台执行,执行如下命令:
$ nohup /home/heiyeluren/php_daemon1.php &

记得最后加上 & 符号,这样才能够跑到后台去运行,执行上述命令后出现如下提示:

[1] 82480
appending output to nohup.out

再回后车后将出现shell提示符。那么上面的提示就是说,所有命令执行的输出信息都会放到 nohup.out 文件中,这个上面已经讲了。然后执行上面命令后,我们每个三十秒在当前目录就会看到多出以test_开头的文件,比如:test_1139901144.txt test_1139901154.txt等等文件,那么就证明我们的程序已经再后台运行了。

那么我们如何终止程序的运行呢?最好办法就是重启操作系统,呵呵,当然,这是不可取的,我们可以使用kill命令来杀掉这个进程,杀进程之前自然后知道进程的PID号,就是Process ID,使用ps命令就能够看到了。

$ ps
  PID  TT  STAT      TIME COMMAND
82374  p3  Ss     0:00.14 -bash (bash)
82510  p3  S      0:00.06 /usr/local/php/bin/php /home/heiyeluren/php_daemon1.php
82528  p3  R+     0:00.00 ps

上面我们已经看到了我们的php的进程id是:82510 ,于是我们再执行kill命令:

$ kill -9 82510
[1]+  Killed                  nohup /home/heiyeluren/php_daemon1.php

看到这么提示就明白这个进程被杀了,再ps,就会发现没有了:

$ ps
  PID  TT  STAT      TIME COMMAND
82374  p3  Ss     0:00.17 -bash (bash)
82535  p3  R+     0:00.00 ps

如果直接ps命令无法看到进程,那么就使用 ps & apos 两个结合命令来查看,一定能够看到进程。
再上面的基础上进程扩展,能够做成属于自己的cron程序,那就不需要cron啦,当然,这只是一种方式