stream_socket_client2

http://news.newhua.com/news1/programming/2007/924/0792411512H35CJ5J4C9G899HA68FJ.html

<?php
echo "Program starts at ". date('h:i:s') . ".\n";

$timeout=10; 
$result=array(); 
$sockets=array(); 
$convenient_read_block=8192;

/* Issue all requests simultaneously; there's no blocking. */
$delay=15;
$id=0;
while ($delay > 0) {
 $s=stream_socket_client("phaseit.net:80", $errno,
  $errstr, $timeout,
  STREAM_CLIENT_ASYNC_CONNECT|STREAM_CLIENT_CONNECT); 
 if ($s) { 
  $sockets[$id++]=$s; 
  $http_message="GET /demonstration/delay?delay=" .
   $delay . " HTTP/1.0\r\nHost: phaseit.net\r\n\r\n"; 
  fwrite($s, $http_message);
 } else { 
  echo "Stream " . $id . " failed to open correctly.";
 } 
 $delay -= 3;


while (count($sockets)) { 
 $read=$sockets; 
 stream_select($read, $w=null, $e=null, $timeout); 
 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, $convenient_read_block); 
   /* A socket is readable either because it has
   data to read, OR because it's at EOF. */
   if (strlen($data) == 0) { 
    echo "Stream " . $id . " closes at " . date('h:i:s') . ".\n";
    fclose($r); 
    unset($sockets[$id]); 
   } else { 
    $result[$id] .= $data; 
   } 
  } 
 } else { 
  /* A time-out means that *all* streams have failed
  to receive a response. */
  echo "Time-out!\n";
  break;
 } 

?>

 

如果运行此清单,您将看到如下所示的输出。

  清单 2. 从清单 1 中的程序获得的典型输出

Program starts at 02:38:50.
Stream 4 closes at 02:38:53.
Stream 3 closes at 02:38:56.
Stream 2 closes at 02:38:59.
Stream 1 closes at 02:39:02.
Stream 0 closes at 02:39:05.


  了解这其中的工作原理至关重要。在较高层次上,第一个程序将发出几个 HTTP 请求并接收 Web 服务器发送给它的页面。虽然生产应用程序将很可能寻找若干个 Web 服务器的地址 —— 可能是 google.com、yahoo.com、ask.com 等 —— 但是此示例将把它的所有请求发送到位于 Phaseit.net 的企业服务器上,只为降低复杂度。

  Web 页面请求在延迟(可变)后返回结果,如下所示。如果程序按顺序发出请求,则需花费大约 15+12+9+6+3 (45) 秒钟才能完成。如清单 2 所示,它实际上花费 15 秒钟完成。性能提高了三倍。

  使这成为可能的是 PHP V5 的新 stream_select 函数。请求都是以常规方法发起,方法为打开几个 stream_socket_client 并向对应于 http://phaseit.net/demonstration/delay?delay=$DELAY 的每个 stream_socket_client 写入 GET。如果您通过浏览器请求此 URL,则在几秒钟之后,您将看到:

Starting at Thu Apr 12 15:05:01 UTC 2007. 
Stopping at Thu Apr 12 15:05:05 UTC 2007. 
4 second delay.


  延迟服务器将作为 CGI 实现,如下所示:

  清单 3. 延迟服务器实现

#!/bin/sh

echo "Content-type: text/html

<HTML> <HEAD></HEAD> <BODY>"

echo "Starting at `date`."
RR=`echo $REQUEST_URI | sed -e 's/.*?//'`
DELAY=`echo $RR | sed -e 's/delay=//'`
sleep $DELAY
echo "<br>Stopping at `date`."
echo "<br>$DELAY second delay.</body></html>"


  虽然清单 3 的特殊实现特定于 UNIX?,但是本文中几乎所有实现都将很好地应用于 Windows?(尤其是 Windows 98 以后的版本)或 PHP 的 UNIX 安装。特别地,清单 1 可以托管在任意一个操作系统中。因此,Linux? 和 Mac OS X 都是 UNIX 变体,因此这里所有的代码都可以在两者的任意一种中运行。

  按照以下顺序向延迟服务器发出请求。

  清单 4. 进程启动顺序

delay=15
delay=12
delay= 9
delay= 6
delay= 3


  stream_select 的作用是尽可能快速地接收结果。在这种情况下,它执行的顺序与发出结果的顺序刚好相反。3 秒后,第一个页面已经准备好读取。程序的这一部分也符合常规 PHP —— 在本例中,使用 fread。就像在其他 PHP 程序一样,读取可以很好地通过 fgets 完成。

  处理将以同样的方法继续。程序将在 stream_select 停止,直至数据就绪。重要的一点是,只要任何 连接具有数据,不管顺序怎样,程序都将开始读取。这是程序进行多任务处理或并发处理来自多个请求的结果的方法。

  注意,这没有对主机 CPU 造成任何负担。经常会遇到这样一些连网程序,以 CPU 使用率急速上升至 100% 的方式在 while 中使用 fread。那种情况不会出现在这里,因为 stream_select 拥有支持立即响应所需的属性(只要有任何读取信息),但是它将在各读取操作间隙的等待时间内产生可忽略的 CPU 负载。

  必备的 stream_select() 知识
 

posted @ 2012-02-16 18:17  taek  阅读(488)  评论(0编辑  收藏  举报