主頁(yè) > 知識(shí)庫(kù) > PHP實(shí)現(xiàn)非阻塞模式的方法分析

PHP實(shí)現(xiàn)非阻塞模式的方法分析

熱門(mén)標(biāo)簽:宿州正規(guī)外呼系統(tǒng)軟件 企業(yè)400電話辦理多少費(fèi)用 桂陽(yáng)公司如何做地圖標(biāo)注 代理打電話機(jī)器人 萍鄉(xiāng)商鋪地圖標(biāo)注 電信外呼系統(tǒng)多少錢(qián)一個(gè)月 太原400電話申請(qǐng)流程 神龍斗士電話機(jī)器人 合肥企業(yè)外呼系統(tǒng)線路

本文實(shí)例講述了PHP實(shí)現(xiàn)非阻塞模式的方法。分享給大家供大家參考,具體如下:

程序非阻塞模式,這里也可以理解成并發(fā)。而并發(fā)又暫且可以分為網(wǎng)絡(luò)請(qǐng)求并發(fā)本地并發(fā) 。

先說(shuō)一下網(wǎng)絡(luò)請(qǐng)求并發(fā)

理論描述

假設(shè)有一個(gè)client,程序邏輯是要請(qǐng)求三個(gè)不同的server,處理各自的響應(yīng)。傳統(tǒng)模型當(dāng)然是順序執(zhí)行,先發(fā)送第一個(gè)請(qǐng)求,等待收到響應(yīng)數(shù)據(jù)后再發(fā)送第二個(gè)請(qǐng)求,以此類(lèi)推。就像是單核CPU,一次只能處理一件事,其他事情被暫時(shí)阻塞。而并發(fā)模式可以讓三個(gè)server同時(shí)處理各自請(qǐng)求,這就可以使大量時(shí)間復(fù)用。

畫(huà)個(gè)圖更好說(shuō)明問(wèn)題:

前者為阻塞模式,忽略請(qǐng)求響應(yīng)等時(shí)間,總耗時(shí)為700ms;而后者非阻塞模式,由于三個(gè)請(qǐng)求可以同時(shí)得到處理,總耗時(shí)只有300ms。

代碼實(shí)現(xiàn)

?php
echo "Program starts at ". date('h:i:s') . "./n";
$timeout = 3;
$sockets = array(); //socket句柄數(shù)組
//一次發(fā)起多個(gè)請(qǐng)求
$delay = 0;
while ($delay++  3)
{
  $sh = stream_socket_client("localhost:80", $errno, $errstr, $timeout,
      STREAM_CLIENT_ASYNC_CONNECT|STREAM_CLIENT_CONNECT);
  /* 這里需要稍微延遲一下,否則下面fwrite中的socket句柄不一定能真正使用
    這里應(yīng)該是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";
  }
}
//非阻塞模式來(lái)接收響應(yīng)
$result = array();
$read_block_size = 8192;
while (count($sockets))
{
  $read = $sockets;
  $n = stream_select($read, $w=null, $e=null, $timeout);
  //if ($n > 0) //據(jù)說(shuō)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);

幾點(diǎn)說(shuō)明:

1、使用stream_socket_client函數(shù)鏈接請(qǐng)求服務(wù)器和端口(簡(jiǎn)便起見(jiàn)這里使用同一地址localhost)。這里不受限于http協(xié)議,可廣泛用于所有TCP/IP協(xié)議。詳細(xì)內(nèi)容請(qǐng)參考手冊(cè)。

2、這里鏈接成功后通過(guò)發(fā)送各自http頭信息來(lái)獲取不同響應(yīng)(這里使用網(wǎng)站根目錄下的test.php做服務(wù)端)。

3、發(fā)送header前需要個(gè)微小的延遲,代碼中已經(jīng)做了注釋。

CLI模式運(yùn)行結(jié)果:

多運(yùn)行幾次會(huì)發(fā)現(xiàn),三次請(qǐng)求結(jié)束順序是無(wú)序的。該demo太過(guò)簡(jiǎn)單導(dǎo)致整個(gè)過(guò)程一秒內(nèi)已完成,但可以針對(duì)三次不同請(qǐng)求做相應(yīng)延遲,來(lái)看出非阻塞時(shí)時(shí)間復(fù)用的效果。

下面再大概說(shuō)下本地并發(fā)

本地并發(fā)只能通過(guò)語(yǔ)言自己的特性在程序本身實(shí)現(xiàn)多任務(wù)效果,一般來(lái)說(shuō)現(xiàn)在的語(yǔ)言會(huì)通過(guò)多線程或多進(jìn)程的方式來(lái)實(shí)現(xiàn)。由于PHP不支持多線程,目前只能采用多進(jìn)程方式,讓操作系統(tǒng)來(lái)幫助實(shí)現(xiàn)本地并發(fā)。

至于代碼實(shí)現(xiàn),可以通過(guò)pcntl擴(kuò)展(封裝fork等進(jìn)程控制函數(shù),和C語(yǔ)言中使用非常相似,windows下不可用)、 proc_openpopen等方式,方法不止一種,這里就不做詳細(xì)介紹了。詳情可自行搜索“php多進(jìn)程”進(jìn)行了解:)

更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專(zhuān)題:《php socket用法總結(jié)》、《php字符串(string)用法總結(jié)》、《PHP數(shù)學(xué)運(yùn)算技巧總結(jié)》、《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門(mén)教程》、《PHP數(shù)組(Array)操作技巧大全》、《PHP數(shù)據(jù)結(jié)構(gòu)與算法教程》、《php程序設(shè)計(jì)算法總結(jié)》及《PHP網(wǎng)絡(luò)編程技巧總結(jié)》

希望本文所述對(duì)大家PHP程序設(shè)計(jì)有所幫助。

您可能感興趣的文章:
  • php 多進(jìn)程編程父進(jìn)程的阻塞與非阻塞實(shí)例分析
  • 簡(jiǎn)單介紹PHP非阻塞模式
  • php使用flock阻塞寫(xiě)入文件和非阻塞寫(xiě)入文件的實(shí)例講解
  • PHP實(shí)現(xiàn)的CURL非阻塞調(diào)用類(lèi)
  • 關(guān)于PHP中協(xié)程和阻塞的一些理解與思考
  • PHP編程中的Session阻塞問(wèn)題與解決方法分析
  • php基于session鎖防止阻塞請(qǐng)求的方法分析
  • 深入解析PHP中的(偽)多線程與多進(jìn)程
  • 深入探究PHP的多進(jìn)程編程方法
  • php多進(jìn)程中的阻塞與非阻塞操作實(shí)例分析

標(biāo)簽:衡陽(yáng) 廊坊 鄂州 綏化 白銀 辛集 崇左 太原

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《PHP實(shí)現(xiàn)非阻塞模式的方法分析》,本文關(guān)鍵詞  PHP,實(shí)現(xiàn),非,阻塞,模式,的,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《PHP實(shí)現(xiàn)非阻塞模式的方法分析》相關(guān)的同類(lèi)信息!
  • 本頁(yè)收集關(guān)于PHP實(shí)現(xiàn)非阻塞模式的方法分析的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章