技术头条 - 一个快速在微博传播文章的方式     搜索本站
您现在的位置首页 --> PHP --> 如何使用PHP编写daemon process

如何使用PHP编写daemon process

浏览:2209次  出处信息

今天下午在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编写而成。

代码实例

下面是这个程序的完整代码:

  1. <?php

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

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

  4. function handle_http_request($address, $port)

  5. {

  6.    $max_backlog =16;

  7.    $res_content ="HTTP/1.1 200 OK

  8.        Content-Length: 15

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

  10.        PHP HTTP Server";

  11.    $res_len = strlen($res_content);

  12. //Create, bind and listen to socket

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

  14. {

  15.        echo "Create socket failed! ";

  16. exit;

  17. }

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

  19. {

  20.        echo "Bind socket failed! ";

  21. exit;

  22. }

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

  24. {

  25.        echo "Listen to socket failed! ";

  26. exit;

  27. }

  28. //Loop

  29. while(TRUE)

  30. {

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

  32. {

  33. continue;

  34. }

  35. else

  36. {

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

  38.            socket_close($accept_socket);

  39. }

  40. }

  41. }

  42. //Run as daemon process.

  43. function run()

  44. {

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

  46. //First child process

  47. {

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

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

  50. //Second child process, which run as daemon.

  51. {

  52. //Replaced with your own domain or address.

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

  54. }

  55. else

  56. {

  57. //First child process exit;

  58. exit;

  59. }

  60. }

  61. else

  62. {

  63. //Wait for first child process exit;

  64.        pcntl_wait($status);

  65. }

  66. }

  67. //Entry point.

  68. run();

  69. ?>

这里我假设各位对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模块相关内容。

检验

下面在命令行下启动这个脚本:

  1. php httpserver.php

用ps命令可以看到我们已经启动了一个daemon进程:

这里我绑定的是我博客的域名www.codinglabs.org,端口是9999,可以按需要进行修改。

下面我先用curl命令看下这个http server是否正常运行:

看来是没问题,再到浏览器中看一下:

结语

当然,这个程序算不上真正的http server,即使作为一个daemon process,也是不完善的,很多必要的事情如修改执行目录(php中可以通过chroot实现)、信号绑定、日志功能等等都没有去做,不过作为一个demo,它已经足够说明php不只是可以编写动态网页处理脚本。如果有的朋友有兴趣,可以使用php将我上面说的功能为这个的http server加上。

还有一点要说明的是,pcntl和sockets模块默认是不安装的,如果在安装php时没有通过参数指定安装,则需要单独安装这两个扩展模块。

而只是一个简单的同步阻塞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进程: image 这里我绑定的是我博客的域名www.codinglabs.org,端口是9999,可以按需要进行修改。 下面我先用curl命令看下这个http server是否正常运行: image 看来是没问题,再到浏览器中看一下: image

结语

   当然,这个程序算不上真正的http server,即使作为一个daemon process,也是不完善的,很多必要的事情如修改执行目录(php中可以通过chroot实现)、信号绑定、日志功能等等都没有去做,不过作为一个demo,它已经足够说明php不只是可以编写动态网页处理脚本。如果有的朋友有兴趣,可以使用php将我上面说的功能为这个的http server加上。 还有一点要说明的是,pcntl和sockets模块默认是不安装的,如果在安装php时没有通过参数指定安装,则需要单独安装这两个扩展模块。


建议继续学习:

  1. 推荐一些socket工具,TCP、UDP调试、抓包工具    (阅读:9434)
  2. 推荐一些socket工具,TCP、UDP调试、抓包工具    (阅读:7273)
  3. 用unix socket加速php-fpm、mysql、redis的连接    (阅读:6514)
  4. 浅析linux kernel network之socket创建    (阅读:5713)
  5. nginx、php-fpm默认配置与性能–TCP socket还是unix domain socket    (阅读:5009)
  6. 如何使用PHP编写daemon process    (阅读:5007)
  7. web socket 心跳包的实现方案    (阅读:4933)
  8. python中的socket代理    (阅读:4820)
  9. netstat和web主机socket文件分析    (阅读:4441)
  10. 使用socket.io和node.js搭建websocket应用    (阅读:4372)
QQ技术交流群:445447336,欢迎加入!
扫一扫订阅我的微信号:IT技术博客大学习
© 2009 - 2024 by blogread.cn 微博:@IT技术博客大学习

京ICP备15002552号-1