對大家推薦很好使用的Linux php系統,像讓大家對Linux php系統有所瞭解,然後對Linux php系統全面講解介紹,希望對大家有用Linux/Unix下守護進程Daemon)大家都知道,比如我們常用的httpd、mysqld等等,就是常駐記憶體啟動並執行程式,類似於Windows下的服務。一般守護進程都是使用C/C++來寫,就是通過fork產生子進程。
當前台shell下的父進程被殺掉,子進程就轉到後台運行,為了不在終端產生輸出資訊,就通過 syslog等函數來寫記錄檔。我們知道php是指令碼語言,通過php的指令碼引擎來執行,所以要做成守護進程比較麻煩。
我們今天就來結合Unix/Linux的命令來實現我們守護進程的功能。Unix中的nohup命令的功能就是不掛斷地運行命令,同時nohup把程式的所有輸出到放到目前的目錄的nohup.out檔案中,如果檔案不可寫,則放到<使用者主目錄>/nohup.out 檔案中。
那麼有了這個命令以後,我們的Linux php程式就寫程shell指令碼。使用迴圈來讓我們的指令碼一直運行,那麼不管我們終端視窗是否關閉,都能夠讓我們的Linux php指令碼一直運行。當然,當我們的Linux php進程被殺或者我們的作業系統重啟了,自然就會中止了。
肯定會問,讓我們的Linux php指令碼做了守護進程又有什麼用處呢?當然有,比如最典型的作用,能夠基本的替代cron的功能,比如我們需要定期實行的某些操作,完全可以交給它來做,不再需要cron。
當然,如果伺服器重啟就沒有辦法了,不過,一般的Unix伺服器不是那麼容易重啟的。另外,我們還可以做一個簡單的伺服器端的功能,比如做一個能夠Telnet過去的伺服器,嘿嘿,可以做成一個小後門,不過這樣實現稍微有點複雜。
例子一:自動組建檔案我們現在來做兩個例子來證明我們上面的說法。首先第一個是每個三十秒自動產生一個檔案,永遠執行下去。首必須確保作業系統是Unix或者Linux,比如可以是FreeBSD、Redhat、Fedora或者SUSE什麼的。
然後我們必須確保我們的Linux 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
上面我們已經看到了我們的Linux 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啦,當然,這隻是一種方式。
例子二:伺服器端的守護進程這個例子跟網路有關,大致就是類比使用Linux php做伺服器端,然後一直後台運行,達到伺服器端Daemon的效果。繼續在我們的主目錄下:/home/heiyeluren,編輯檔案php_daemon2.php:
- $ vi php_daemon2.php
- 輸入如下代碼代碼來自PHP手冊,我進行了修改注釋):
-
- #! /usr/local/php/bin/php
- <?php
- /* 設定不顯示任何錯誤 */
- 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 = "==========================================\r\n" .
- " Welcome to the PHP Test Server. \r\n\r\n".
- " To quit, type 'quit'. \r\n" .
- " To shut down the server type 'shutdown'.\r\n" .
- " To get help message type 'help'.\r\n" .
- "==========================================\r\n" .
- "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 \r\n\r\n".
- " To quit, type 'quit'. \r\n" .
- " To shut down the server type 'shutdown'.\r\n" .
- " To get help message type 'help'.\r\n" .
- "php> ";
- socket_write($msgsock, $msg, strlen($msg));
- continue;
- }
- /* 用戶端輸入命令不存在時提示資訊 */
- $talkback = "PHP: unknow command '$buf'.\r\nphp> ";
- socket_write($msgsock, $talkback, strlen($talkback));
- echo "$buf\n";
- } 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> asdf
- PHP: unknow command 'asdf'.
-
執行help命令可以擷取協助:
- php> help
- PHP Server Help Message
-
- To quit, type 'quit'.
- To shut down the server type 'shutdown'.
- To get help message type 'help'.
這個伺服器端就不介紹了,可以自行擴充。殺進程跟例子一類似。通過以上學習,我們知道Linux php也可以做守護進程,如果設計的好,功能也會比較強大,不過我們這裡只是學習而已,可以自行研究更新。本文參考了Linux php中文手冊,多看手冊,對自己非常有好處。
- Linux PHP編譯產生擴充與修改配置
- Linux pam_mysql的配置介紹伺服器程式
- Linux JDK配置 Java 編譯、運行環境
- Linux磁碟分割的操作工具與命令使用
- Linux Kernel核心2.6.33正式版發布