程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> PHP編程 >> 關於PHP編程 >> 如何使用PHP編寫daemon process

如何使用PHP編寫daemon process

編輯:關於PHP編程

 

今天下午在segmentfault.com看到一個提問,提問標題是“PHP怎麼做服務化”,其中問道php是不是只能以web方式調用。其實很多人對PHP的使用場景都有誤解,認為php只能用於編寫web腳本,實際上,從PHP4開始,php的使用場景早已不限於處理web請求。

從php的架構體系來說,php分為三個層次:sapi、php core和zend engine。php core本身和web沒有任何耦合,php通過sapi與其它應用程序通信,例如mod_php就是為apache編寫的sapi實現,同樣,fpm是一個基於fastcgi協議的sapi實現,這些sapi都是與web server配合用於處理web請求的。但是也有許多sapi與web無關,例如cli sapi可以使得在命令行環境下直接執行php,embed sapi可以將php嵌入其它語言(如Lua)那樣。這裡我並不打算詳細討論php的架構體系和sapi的話題,只是說明從架構體系角度目前的php早已被設計為支持各種環境,而非為web獨有。

除了架構體系的支持外,php豐富的擴展模塊也為php在不同環境發揮作用提供了後盾,例如本文要提到的pcntl模塊和posix模塊配合可以實現基本的進程管理、信號處理等操作系統級別的功能,而sockets模塊可以使php具有socket通信的能力。因此php完全可以用於編寫類似於shell或perl常做的工具性腳本,甚至是具有server性質的daemon process。

為了展示php如何編寫daemon server,我用php編寫了一個簡單的http server,這個server以daemon process的形式運行。當然,為了把重點放在如何使用php編寫daemon,我沒有為這個http server實現具體業務邏輯,但它可以監聽指定端口,接受http請求並返回給客戶端一條固定的文本,整個過程通過socket實現,全部由php編寫而成。

代碼實例

下面是這個程序的完整代碼:

 

<?php

 

//Accpet the http client request and generate response content.

//As a demo, this function just send "PHP HTTP Server" to client.

function handle_http_request($address, $port)

{

    $max_backlog = 16;

    $res_content = "HTTP/1.1 200 OK

Content-Length: 15

Content-Type: text/plain; charset=UTF-8

 

PHP HTTP Server";

    $res_len = strlen($res_content);

 

    //Create, bind and listen to socket

    if(($socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP)) === FALSE)

    {

        echo "Create socket failed!\n";

        exit;

    }   

 

    if((socket_bind($socket, $address, $port)) === FALSE)

    {

        echo "Bind socket failed!\n";

        exit;

    }

 

    if((socket_listen($socket, $max_backlog)) === FALSE)

    {

        echo "Listen to socket failed!\n";

        exit;

    }

 

    //Loop

    while(TRUE)

    {

        if(($accept_socket = socket_accept($socket)) === FALSE)

        {

            continue;

        }

        else

        {

            socket_write($accept_socket, $res_content, $res_len);

            socket_close($accept_socket);

        }

    }

}

 

//Run as daemon process.

function run()

{

    if(($pid1 = pcntl_fork()) === 0)

    //First child process

    {

        posix_setsid(); //Set first child process as the session leader.

 

        if(($pid2 = pcntl_fork()) === 0)

        //Second child process, which run as daemon.

        {

            //Replaced with your own domain or address.

            handle_http_request('www.codinglabs.org', 9999);

        }

        else

        {

            //First child process exit;

            exit;

        }

    }

    else

    {

        //Wait for first child process exit;

        pcntl_wait($status);

    }

}

 

//Entry point.

run();

 

?>

 

這裡我假設各位對Unix環境編程都比較了解,所以不做太多細節的解釋,只梳理一下。簡單來看,這個程序主要由兩個部分組成,handle_http_request函數負責處理http請求,其編寫方法與用C編寫的tcp server類似:創建socket、綁定、監聽,然後通過一個循環處理每個connect過來的客戶端,一旦accept到一個連接,則輸出固定的文本“PHP HTTP Server”(當然http頭需要首先構建好),這裡沒有考慮多路復用和非阻塞等情況,而只是一個簡單的同步阻塞tcp server。

run函數負責將整個程序變為daemon process,方法和Unix環境下C的方法很類似,通過兩次fork,第一次fork後調用setsid將子進程1變為session leader,這樣就可以讓子進程2與其祖先detach,即使祖先進程結束了它也會繼續運行(托孤給init進程)。相關細節我不再贅述,對Unix進程相關不熟悉的朋友可以參考《Advanced Programming in the UNIX Environment》一書。

注意,在這裡pcntl_fork對應Unix中的fork,pcntl_wait對應wait,而posix_setsid對應setsid,更多函數可以參考PHP Manual中的pcntl和fork模塊相關內容。

檢驗

下面在命令行下啟動這個腳本:

 

php httpserver.php  

 

 

用ps命令可以看到我們已經啟動了一個daemon進程:   1 image

這裡我綁定的是我博客的域名www.codinglabs.org,端口是9999,可以按需要進行修改。

下面我先用curl命令看下這個http server是否正常運行:

image

看來是沒問題,再到浏覽器中看一下:

image

結語

當然,這個程序算不上真正的http server,即使作為一個daemon process,也是不完善的,很多必要的事情如修改執行目錄(php中可以通過chroot實現)、信號綁定、日志功能等等都沒有去做,不過作為一個demo,它已經足夠說明php不只是可以編寫動態網頁處理腳本。如果有的朋友有興趣,可以使用php將我上面說的功能為這個的http server加上。

還有一點要說明的是,pcntl和sockets模塊默認是不安裝的,如果在安裝php時沒有通過參數指定安裝,則需要單獨安裝這兩個擴展模塊。

 

摘自 codinglabs.org

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved