3

这让我抓狂。我必须在 php 5.3 中创建一个非常简单的非阻塞套接字脚本,其中客户端连接到服务器,两者都使用非阻塞套接字。

我已经尝试过phpsocketdaemonphp 手册中的示例,但是在这两种情况下,当我尝试连接到服务器时,我都会收到以下错误:

socket_connect() [function.socket-connect]: unable to connect [10035]:
A non-blocking socket operation could not be completed immediately

我发生错误的客户端脚本:

$service_port = 2002;
$address = '127.0.0.1';

$socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
socket_set_nonblock($socket); 
$result = socket_connect($socket, $address, $service_port);
...

我在 Win 7 上使用 Zend Server 5.6.0 SP1 的免费版本。

有谁知道如何解决这个问题或知道一个简单易懂的非阻塞套接字客户端/服务器脚本示例?

4

2 回答 2

10

当您将套接字设置为非阻塞时,您不能期望socket_connect()如果连接的结果返回 TRUE,否则返回 FALSE。

PHP手册页

如果套接字是非阻塞的,则此函数返回 FALSE 并显示错误 Operation now in progress。

在任何语言中都是如此。您必须设置套接字“阻塞”,或者您必须在检查您是否正确连接之前轮询/选择文件描述符。在 PHP 中,您可能会socket_connect()在一小段时间后调用该函数,以检查它是否返回 true、false 或等待超时到期。

试试这个代码[编辑修复超时例程的一个小错误]:

<?php

  $service_port = 2002;
  $address = '127.0.0.1';
  $timeout = 3;

  $socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
  socket_set_nonblock($socket);
  $error = NULL;
  $attempts = 0;
  $timeout *= 1000;  // adjust because we sleeping in 1 millisecond increments
  $connected = FALSE;
  while (!($connected = @socket_connect($socket, $address, $service_port)) && ($attempts++ < $timeout)) {
        $error = socket_last_error();
        if ($error != SOCKET_EINPROGRESS && $error != SOCKET_EALREADY) {
              echo "Error Connecting Socket: ".socket_strerror($error) . "\n";
              socket_close($socket);
              return NULL;
        }
        usleep(1000);
  }

  if (!$connected) {
        echo "Error Connecting Socket: Connect Timed Out After " . $timeout/1000 . " seconds. ".socket_strerror(socket_last_error()) . "\n";
        socket_close($socket);
        return NULL;
  }

?>
于 2012-03-18T19:47:22.767 回答
3

以前的解决方案对我不起作用,所以我使用 socket_select 找到了我的解决方案:

<?php
$service_port = 80;
$address = '127.0.0.1';
$timeout = 3;

$socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
socket_set_nonblock($socket);
$error = NULL;
$attempts = 0;
$timeout *= 1000;  // adjust because we sleeping in 1 millisecond increments
$connected = FALSE;
$connected = @socket_connect($socket, $address, $service_port);
if (!$connected)
{
    $error = socket_last_error();
    if ($error != 10035 && $error != SOCKET_EINPROGRESS && $error != SOCKET_EALREADY) {
        echo "Error Connecting Socket: ".socket_strerror($error) . "\n";
        socket_close($socket);
        exit();
    }
}
$writables = array();
$writables[] = $socket;
$readable = array();
$e = array();
$result = socket_select($readable, $writables, $e, $timeout);
if (!$result)
    die("Unable to connect to socket: Timeout");
/* blablah send lots of things */
socket_close($socket);

它适用于 Windows 上的 XAMPP 和我的 linux 服务器。

于 2014-03-31T09:58:35.610 回答