在PHP中使用无限循环的POST请求,不断发送一些数据点(每秒瞄准一次)。数据高度依赖于时间。如果超时,我不关心丢失的数据,我只想快速转到下一个数据点。
我最初使用file_get_contents()
(超时值为1)来执行此POST请求,但我遇到了一些应该超时的问题而不是。我读了很多关于这个的问题,普遍的共识是file_get_contents()
上的超时没有被触发,因为它是读超时,而不是 connect 超时。
我还读到套接字有两种超时,所以我改为使用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秒的挂起。任何帮助表示赞赏。
答案 0 :(得分:3)
我最终在这方面取得了一些进展。我从未发现错误,但怀疑它可能只是在设备的过载硬件上阻塞。为了解决这个问题,我创建了一个PHP文件来执行迭代一次,并在其上设置了一秒的时间限制。然后我有另一个无限循环的文件,并不断请求第一个文件。理想情况下,如果迭代文件花费的时间太长,请求将失败并且循环继续,这足够接近我想要的。这有点黑客,但它似乎消除了这个问题。下面是一个非常概括的例子。
迭代:
<?php
set_time_limit(1);
// fsockopen() etc.
?>
循环:
<?php
$i = 0;
for($i = 0; ; $i++)
{
file_get_contents("localhost/iteration_file.php");
}
?>
也许这会对某人有所帮助。最终,我不得不切换到cURL,但这种方法在很长一段时间内都很有用,并且非常可靠。
注意:此答案反映了我实际执行的操作,并且不声称能够实现最佳实践。我不知道很多最佳实践。您还可以使用include伪指令包含另一个文件,或将功能分解为方法。
几个月后编辑:我打算将此标记为答案,因为它在很大程度上解决了我当时的问题。