PHP实现非阻塞模式的方法分析


Posted in PHP onJuly 26, 2018

本文实例讲述了PHP实现非阻塞模式的方法。分享给大家供大家参考,具体如下:

程序非阻塞模式,这里也可以理解成并发。而并发又暂且可以分为网络请求并发本地并发

先说一下网络请求并发

理论描述

假设有一个client,程序逻辑是要请求三个不同的server,处理各自的响应。传统模型当然是顺序执行,先发送第一个请求,等待收到响应数据后再发送第二个请求,以此类推。就像是单核CPU,一次只能处理一件事,其他事情被暂时阻塞。而并发模式可以让三个server同时处理各自请求,这就可以使大量时间复用。

画个图更好说明问题:

PHP实现非阻塞模式的方法分析

前者为阻塞模式,忽略请求响应等时间,总耗时为700ms;而后者非阻塞模式,由于三个请求可以同时得到处理,总耗时只有300ms。

代码实现

<?php
echo "Program starts at ". date('h:i:s') . "./n";
$timeout = 3;
$sockets = array(); //socket句柄数组
//一次发起多个请求
$delay = 0;
while ($delay++ < 3)
{
  $sh = stream_socket_client("localhost:80", $errno, $errstr, $timeout,
      STREAM_CLIENT_ASYNC_CONNECT|STREAM_CLIENT_CONNECT);
  /* 这里需要稍微延迟一下,否则下面fwrite中的socket句柄不一定能真正使用
    这里应该是PHP的一处bug,查了一下,官方bug早在08年就有人提交了
    我的5.2.8中尚未解决,不知最新的5.3中是否修正
  */
  usleep(10);
  if ($sh) {
    $sockets[] = $sh;
    $http_header = "GET /test.php?n={$delay} HTTP/1.0/r/n";
    $http_header .= "Host: localhost/r/n";
    $http_header .= "Accept: */*/r/n";
    $http_header .= "Accept-Charset: */r/n";
    $http_header .= "/r/n";
    fwrite($sh, $http_header);
  } else {
    echo "Stream failed to open correctly./n";
  }
}
//非阻塞模式来接收响应
$result = array();
$read_block_size = 8192;
while (count($sockets))
{
  $read = $sockets;
  $n = stream_select($read, $w=null, $e=null, $timeout);
  //if ($n > 0) //据说stream_select返回值不总是可信任的
  if (count($read))
  {
    /* stream_select generally shuffles $read, so we need to
      compute from which socket(s) we're reading. */
    foreach ($read as $r)
    {
      $id = array_search($r, $sockets);
      $data = fread($r, $read_block_size);
      if (strlen($data) == 0)
      {
        echo "Stream {$id} closes at " . date('h:i:s') . "./n";
        fclose($r);
        unset($sockets[$id]);
      } else {
        if (!isset($result[$id])) $result[$id] = '';
        $result[$id] .= $data;
      }
    }
  } else {
    echo "Time-out!/n";
    break;
  }
}
//print_r($result);

几点说明:

1、使用stream_socket_client函数链接请求服务器和端口(简便起见这里使用同一地址localhost)。这里不受限于http协议,可广泛用于所有TCP/IP协议。详细内容请参考手册。

2、这里链接成功后通过发送各自http头信息来获取不同响应(这里使用网站根目录下的test.php做服务端)。

3、发送header前需要个微小的延迟,代码中已经做了注释。

CLI模式运行结果:

PHP实现非阻塞模式的方法分析

多运行几次会发现,三次请求结束顺序是无序的。该demo太过简单导致整个过程一秒内已完成,但可以针对三次不同请求做相应延迟,来看出非阻塞时时间复用的效果。

下面再大概说下本地并发

本地并发只能通过语言自己的特性在程序本身实现多任务效果,一般来说现在的语言会通过多线程或多进程的方式来实现。由于PHP不支持多线程,目前只能采用多进程方式,让操作系统来帮助实现本地并发。

至于代码实现,可以通过pcntl扩展(封装fork等进程控制函数,和C语言中使用非常相似,windows下不可用)、 proc_openpopen等方式,方法不止一种,这里就不做详细介绍了。详情可自行搜索“php多进程”进行了解:)

希望本文所述对大家PHP程序设计有所帮助。

PHP 相关文章推荐
PHP脚本的10个技巧(7)
Oct 09 PHP
php 上一篇,下一篇文章实现代码与原理说明
May 09 PHP
探讨php中遍历二维数组的几种方法详解
Jun 08 PHP
PHP Class&amp;Object -- PHP 自排序二叉树的深入解析
Jun 25 PHP
php数组删除元素示例
Mar 21 PHP
ThinkPHP3.1.3版本新特性概述
Jun 19 PHP
CI操作cookie的方法分析(基于helper类库)
Mar 28 PHP
php封装的数据库函数与用法示例【参考thinkPHP】
Nov 08 PHP
php微信开发之百度天气预报
Nov 18 PHP
解决微信授权回调页面域名只能设置一个的问题
Dec 11 PHP
关于PHP中协程和阻塞的一些理解与思考
Aug 11 PHP
php5对象复制、clone、浅复制与深复制实例详解
Aug 14 PHP
php实现等比例压缩图片
Jul 26 #PHP
PHP输出Excel PHPExcel的方法
Jul 26 #PHP
PHP微信H5支付开发实例
Jul 25 #PHP
php app支付宝回调(异步通知)详解
Jul 25 #PHP
php支付宝APP支付功能
Jul 29 #PHP
PHP多个图片压缩成ZIP的方法
Aug 18 #PHP
PHP上传文件及图片到七牛的方法
Jul 25 #PHP
You might like
如何让thinkphp在模型中自动完成session赋值小教程
2014/09/05 PHP
php中get_object_vars()方法用法实例
2015/02/08 PHP
PHP封装的多文件上传类实例与用法详解
2017/02/07 PHP
gearman中worker常驻后台,导致MySQL server has gone away的解决方法
2020/02/27 PHP
基于jQuery试卷自动排版系统
2010/07/18 Javascript
JavaScript 对任意元素,自定义右键菜单的实现方法
2013/05/08 Javascript
js左侧三级菜单导航实例代码
2013/09/13 Javascript
innerText 使用示例
2014/01/23 Javascript
悬浮数字的实现案例
2014/02/19 Javascript
JavaScript实现数字数组正序排列的方法
2015/04/06 Javascript
$.extend 的一个小问题
2015/06/18 Javascript
javascript实现下班倒计时效果的方法(可桌面通知)
2015/07/10 Javascript
JS原生数据双向绑定实现代码
2017/08/14 Javascript
vue中的scope使用详解
2017/10/29 Javascript
js中bool值的转换及“&amp;&amp;”、“||”、 “!!”详解
2017/12/21 Javascript
vue-cli3 配置开发与测试环境详解
2019/05/17 Javascript
微信小程序 高德地图路线规划实现过程详解
2019/08/05 Javascript
微信小程序页面调用自定义组件内的事件详解
2019/09/12 Javascript
JavaScript使用localStorage存储数据
2019/09/25 Javascript
小程序跨页面交互的作用与方法详解
2020/01/07 Javascript
在vue中使用eslint,配合vscode的操作
2020/11/09 Javascript
Python数据分析之真实IP请求Pandas详解
2016/11/18 Python
Canvas绘制浮动球效果的示例
2017/12/29 HTML / CSS
奥地利时尚、美容、玩具和家居之家:Kastner & Öhler
2020/04/26 全球购物
会计实习生工作总结的自我评价
2013/10/07 职场文书
《只有一个地球》教学反思
2014/02/14 职场文书
干部下基层实施方案
2014/03/14 职场文书
销售员岗位职责
2014/06/09 职场文书
销售目标责任书
2014/07/23 职场文书
退休欢送会致辞
2015/07/31 职场文书
孩子满月酒答谢词
2015/09/30 职场文书
谢师宴家长答谢词
2015/09/30 职场文书
如何使JavaScript休眠或等待
2021/04/27 Javascript
Golang全局变量加锁的问题解决
2021/05/08 Golang
关于mysql中时间日期类型和字符串类型的选择
2021/11/27 MySQL
python中mongodb包操作数据库
2022/04/19 Python