fsockopen连接在超时前不会关闭

时间:2011-07-04 16:01:47

标签: php fgets fsockopen

背景: 我必须创建一个普通站点,接受传入的发布XML并通过套接字连接将XML发送到服务器,然后显示从服务器发回的XML。容易腻。

问题: 我使用 fsockopen()连接到服务器并发送XML没有问题。从服务器读取XML是一个全新的问题。正常

while (!feof($fp)) {    
    echo fgets($fp);
}

没有做到这一点,因为服务器返回一个XML字符串,只返回一个XML字符串(没有长度信息,eof,eol等)。因此,它会等到超时被触发,显示收到的XML和超时错误。我的问题类似于dinosaur

简而言之,我想在套接字上读取XML,并在不再发送数据时立即关闭它(不等待超时)。将超时设置为较低值也是不可行的,因为服务器响应可能在2-30秒之间变化。

解决方案: 在整个下午挣扎之后,我决定为这个问题分享以下解决方案(批评)。

$fp = fsockopen("123.456.789.1", 80, $errno, $errstr, 30);
if (!$fp) {
    echo "$errstr ($errno)";
} else {
    $wait = true;    
    $out = '<samplexml><item>something</item></samplexml>';

    // [1] disable blocking
    stream_set_blocking($fp, 0);
    fwrite($fp, $out);

    while (!feof($fp)) {
        $r = fgets($fp);
        echo $r;
        if (!strcmp($r, "")){                
            if (!$wait) {
                // [2] has recieved data on this socket before
                break;
            }
        } else {
            $wait = false;
        }
    }
    fclose($fp);
}

事实证明,我的主要问题是阻止。首先,[1]我必须禁用stream_set_blocking,以便 fgets()可以不断检查新数据是否可用。如果没有禁用 fgets()将从服务器获取XML,然后循环将在第二次尝试时卡住,因为它将等待更多数据可用(这将永远不会)。

我知道,只要我们读取了一些数据,我们就可以立即关闭连接,如果返回任何空的 fgets()(因此我们仍然可以设置 fgets的第二个参数()如果有必要的话。)

使用此网站数月后,我终于在stackoverflow上发布了一些内容。

3 个答案:

答案 0 :(得分:2)

该代码段有几个问题:

  • 首先,比较strcmp($r, "")实际上没有意义。它有效,但没有意义。我不认为fgets会返回一个空字符串。如果没有更多可用数据,则会返回 FALSE 。您的比较工作的原因是FALSE转换为空字符串。但它会使你的代码不清楚。
  • 其次,由于您在非阻塞模式下拥有流,因此数据暂时无法使用。通过在fgets返回false后立即断开循环,您可能只读取操作系统缓存的第一批数据。你试图通过强制等待没有收到数据来解决这个问题,但这是一个脆弱的解决方案。
  • 最后,你忙着循环等待数据。这是非常耗费资源的。

使用stream_select可以解决最后两点。这样,您可以强制执行大的超时,直到收到第一个数据包,然后使用一个小的超时。

答案 1 :(得分:0)

您可以使用fread()并一次拉入一个字节。非常低效,但您可以解析响应,并检测XML何时结束,而不必等待连接超时。在hacky伪代码中:

while(true) {
    $char = fread($fp, 1);
    $xml .= $char;
    if (is_complete($xml)) {
         break;
    }
}

答案 2 :(得分:0)

我使用套接字。我发送一个xml字符串,我也收到一个xml字符串。当服务器向我发送字符串'</response>'时,我必须关闭套接字。在下面你可以找到我的代码,如果这可以帮助某人:

            stream_set_blocking($fp, 0);
            fwrite($fp, $in);
            $result = '';

            while (!feof($fp)) {
                $r = fgets($fp);
                $result .= $r;
                //echo $r;
                if (strstr($r, "</response>") != null)
                    break;
            }
            fclose($fp);