PHP編寫daemon process
今天下午在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 OKContent-Length: 15Content-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到一個串連...
感謝閱讀,希望能協助到大家,謝謝大家對本站的支援!