fsockopen() 超时有时不起作用

4
不断地发送一些数据点(目标是每秒一次),使用 PHP 中的 POST 请求的无限循环。这些数据高度依赖于时间。如果出现超时,我不关心丢失的数据,只想快速转移到下一个数据点。
最初,我使用了 file_get_contents()(超时值为1)来进行此 POST 请求,但我遇到了一些问题,本应该超时却没有。我阅读了很多相关问题,普遍认为 file_get_contents() 上的超时未被触发,因为它是一个“读取”超时,而不是“连接”超时。
我还了解到套接字有两种超时类型,所以我将代码更改为使用 fsockopen()。以下是代码片段。
    // waaaay up at the top of the file
    ini_set("default_socket_timeout", 1); // set default socket connect timeout
    //...
    // inside the loop
    echo("4...");
    $url = 'othersite.ie';
    $fp = fsockopen($url, 80, $errno, $errstr, 1); // connect timeout of 1
    if(!$fp)
    {
        echo($errno .": ". $errstr);
    }
    else
    {
        echo("5...");
        stream_set_timeout($fp, 1, 0); // read timeout of 1

        $content = http_build_query($data); // array assembled before this snippet               
        fwrite($fp, "POST /some/address/at/url HTTP/1.1\r\n");
        fwrite($fp, "Host: www.examplesite.ie\r\n");
        fwrite($fp, "Content-Type: application/x-www-form-urlencoded\r\n");
        fwrite($fp, "Content-Length: ".strlen($content)."\r\n");
        fwrite($fp, "Connection: close\r\n");
        fwrite($fp, "\r\n");

        fwrite($fp, $content);

        while(!feof($fp))
        {
            $result = $result . fgets($fp, 1024);
        }
        echo("6...");

        if (empty($result))
        {
            echo("No result");
        }
    }

然而,这似乎没有起到任何帮助。我添加了一些echo语句,以便如果我观察输出,我可以看到延迟发生的位置,而且它一直在“4...”和“5...”之间变化。每个延迟大约十秒钟,它们似乎没有特定的顺序。大多数情况下(超过99%),连接正常,读取正常,我将数据发送到服务器。但至少要试图消除那些10秒钟的停顿。任何帮助都将不胜感激。

1个回答

3
我最终在这个问题上取得了一些进展。我没有找到错误,但怀疑可能是设备过载的硬件阻塞。为了解决这个问题,我创建了一个PHP文件来执行迭代,并在其上设置了一秒钟的时间限制。然后我又创建了另一个文件,它会无限循环并不断请求第一个文件。理想情况下,如果迭代文件花费太长时间,请求将失败并继续循环,这已经足够接近我想要的效果了。虽然有点hacky,但它似乎消除了问题。下面是一个极度概括的例子。
迭代:
<?php

    set_time_limit(1);
    // fsockopen() etc.
?>

循环:

<?php
    $i = 0;
    for($i = 0; ; $i++)
    {
        file_get_contents("localhost/iteration_file.php");
    }
?>

也许这会帮助到某些人。最终,我还是需要转换到cURL,但是这种方法很有用了很长一段时间,并且非常可靠。
注:这个答案反映了我实际所做的事情,并且并不声称符合最佳实践。我不知道很多最佳实践。您也可以使用include指令来包含另一个文件,或将功能分解为方法。
几个月后的编辑:我会将其标记为答案,因为它在当时大部分解决了我的问题。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接