背景: 我必须创建一个普通站点,接受传入的发布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上发布了一些内容。
答案 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);