Linux php系統的起源原理功能時間
對大家推薦很好使用的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中文手冊,多看手冊,對自己非常有好處。
【編輯推薦】