PHP守護Linux/Unix進程
[ 起源 ]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] 82480appending output to nohup.out再回后車后將出現shell提示符。那么上面的提示就是說,所有命令執行的輸出信息都會放到 nohup.out文件中,這個上面已經講了。然后執行上面命令后,我們每個三十秒在當前目錄就會看到多出以test_開頭的文件,比如:test_1139901144.txt test_1139901154.txt等等文件,那么就證明我們的程序已經再后臺運行了。那么我們如何終止程序的運行呢?最好辦法就是重啟操作系統,呵呵,當然,這是不可取的,我們可以使用kill命令來殺掉這個進程,殺進程之前自然后知道進程的PID號,就是Process ID,使用ps命令就能夠看到了。$ psPID TT STAT TIME COMMAND82374 p3 Ss 0:00.14 -bash (bash)82510 p3 S 0:00.06 /usr/local/php/bin/php /home/heiyeluren/php_daemon1.php82528 p3 R+ 0:00.00 ps上面我們已經看到了我們的php的進程id是:82510 ,于是我們再執行kill命令:$ kill -9 82510[1]+ Killed nohup /home/heiyeluren/php_daemon1.php看到這么提示就明白這個進程被殺了,再ps,就會發現沒有了:$ psPID TT STAT TIME COMMAND82374 p3 Ss 0:00.17 -bash (bash)82535 p3 R+ 0:00.00 ps如果直接ps命令無法看到進程,那么就使用 ps & apos 兩個結合命令來查看,一定能夠看到進程。再上面的基礎上進程擴展,能夠做成屬于自己的cron程序,那就不需要cron啦,當然,這只是一種方式 例子二:服務器端的守護進程這個例子跟網絡有關,大致就是模擬使用php做服務器端,然后一直后臺運行,達到服務器端Daemon的效果。繼續在我們的主目錄下:/home/heiyeluren,編輯文件php_daemon2.php:$ vi php_daemon2.php輸入如下代碼(代碼來自PHP手冊,我進行了修改注釋):#! /usr/local/php/bin/php<?php/* http://www.mypchelp.cn/php.asp 設置不顯示任何錯誤 */error_reporting(0);/* 腳本超時為無限 */set_time_limit(0);/* 開始固定清除 */ob_implicit_flush();/* 本機的IP和需要開放的端口 */$address = ’192.168.0.1’;$port = 10000;/* 產生一個Socket */if (($sock = socket_create(AF_INET, SOCK_STREAM, SOL_TCP)) < 0) {echo “socket_create() failed: reason: ” . socket_strerror($sock) . “n”;}/* 把IP地址端口進行綁定 */if (($ret = socket_bind($sock, $address, $port)) < 0) {echo “socket_bind() failed: reason: ” . socket_strerror($ret) . “n”;}
/* 監聽Socket連接 */if (($ret = socket_listen($sock, 5)) < 0) {echo “socket_listen() failed: reason: ” . socket_strerror($ret) . “n”;}/* 永遠循環監接受用戶連接 */do {if (($msgsock = socket_accept($sock)) < 0) {echo “socket_accept() failed: reason: ” . socket_strerror($msgsock) . “n”;break;}/* 發送提示信息給連接上來的用戶 */$msg = “==========================================rn” .” Welcome to the PHP Test Server. rnrn”.” To quit, type ’quit’. rn” .” To shut down the server type ’shutdown’.rn” .” To get help message type ’help’.rn” .”==========================================rn” .”php> “;socket_write($msgsock, $msg, strlen($msg));do {if (false === ($buf = socket_read($msgsock, 2048, PHP_NORMAL_READ))) {echo “socket_read() failed: reason: ” . socket_strerror($ret) . “n”;break 2;}if (!$buf = trim($buf)) {continue;}/* 客戶端輸入quit命令時候關閉客戶端連接 */if ($buf == ’quit’) {break;}/* 客戶端輸入shutdown命令時候服務端和客戶端都關閉 */if ($buf == ’shutdown’) {socket_close($msgsock);break 2;}/* 客戶端輸入help命令時候輸出幫助信息 */if ($buf == ’help’) {$msg = ” PHP Server Help Message rnrn”.” To quit, type ’quit’. rn” .” To shut down the server type ’shutdown’.rn” .” To get help message type ’help’.rn” .”php> “;socket_write($msgsock, $msg, strlen($msg));continue;}/* 客戶端輸入命令不存在時提示信息 */$talkback = “PHP: unknow command ’$buf’.rnphp> “;socket_write($msgsock, $talkback, strlen($talkback));echo “$bufn”;} while (true);socket_close($msgsock);} while (true);/* 關閉Socket連接 */socket_close($sock);?>保存以上代碼退出。上面的代碼大致就是完成一個類似于Telnet服務器端的功能,就是當服務器端運行該程序的時候,客戶端能夠連接該服務器的10000端口進行通信。加上文件的可執行權限:$ chmod +x /home/heiyeluren/php_daemon2.php在服務器上執行命令:$ nohup /home/heiyeluren/php_daemon2.php &就進入了后臺運行,我們通過Windows的客戶端telnet上去:C:>telnet 192.168.0.1 10000如果提示:正在連接到192.168.0.188…不能打開到主機的連接, 在端口 10000: 連接失敗則說明服務器端沒有開啟,或者上面的程序沒有正確執行,請檢查php是否 –enable-sockets 功能。如果提示:==========================================Welcome to the PHP Test Server.To quit, type ’quit’.To shut down the server type ’shutdown’.To get help message type ’help’.==========================================php>則說明順利連接上了我們的PHP寫的服務器端守護進程,在php>提示符后面能夠執行help、quit、shutdown等三個命令,如果命令輸入不是這三個,則提示:php> asdfPHP: unknow command ’asdf’.執行help命令可以獲取幫助php> helpPHP Server Help MessageTo quit, type ’quit’.To shut down the server type ’shutdown’.To get help message type ’help’.這個服務器端就不介紹了,可以自行擴展。殺進程跟例子一類似。[ 總結 ]通過以上學習,我們知道php也可以做守護進程,如果設計的好,功能也會比較強大,不過我們這里只是學習而已,可以自行研究更新。本文參考了php中文手冊,多看手冊,對自己非常有好處。