7

我有一个 php 脚本,可以从带有验证的 csv 文件中导入大数据。
为此,我需要向用户显示进度。我为此使用了事件流。
当我回显某些内容时,我希望将其一一传输到客户端,而不是服务器批量发送整个输出。
我已经玩过 ob_start()、ob_implicit_flush() 和 ob_flush(),但它们没有用。
我的脚本在另一台服务器上运行良好。下面给出服务器配置:

代码未按预期响应的服务器配置,即

操作系统:Linux
PHP 版本 5.4.36-0+deb7u3
服务器 API:CGI/FastCGI
内存限制:128M
output_buffering:没有值

正如我所说,代码在另一台具有几乎相同配置的服务器上正常工作,即

操作系统:Linux
PHP 版本 5.4.37
服务器 API:CGI/FastCGI
内存限制:256MB
output_buffering:没有值

下面是我发送事件的示例代码:

<?php
header("Content-Type: text/event-stream");
header("Cache-Control: no-cache");
header("Access-Control-Allow-Origin: *");

$lastEventId = floatval(isset($_SERVER["HTTP_LAST_EVENT_ID"]) ? $_SERVER["HTTP_LAST_EVENT_ID"] : 0);
if ($lastEventId == 0) {
    $lastEventId = floatval(isset($_GET["lastEventId"]) ? $_GET["lastEventId"] : 0);
}

echo ":" . str_repeat(" ", 2048) . "\n"; // 2 kB padding for IE
echo "retry: 2000\n";

// event-stream
$i = $lastEventId;

while ($i <= 100) {
    if($i==100){
        echo "data: stop\n";
        ob_flush();
        flush();
        break;
    } else {
        echo "id: " . $i . "\n";
        echo "data: " . $i . ";\n\n";
        ob_flush();
        flush();
        sleep(1);
    }
    $i++;
}
?>

以下是我需要回复的客户页面:

<!DOCTYPE html>
<html>

<head>
    <meta charset="utf-8" />
    <title>EventSource example</title>
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <script src="../jquery/eventsource.js"></script>
    <script>
        var es = new EventSource("events.php");
        var listener = function(event) {
            console.log(event.data);
            var type = event.type;
            if (event.data == 'stop') {
                es.close();
            } else {
                var div = document.createElement("div");
                div.appendChild(document.createTextNode(type + ": " + (type === "message" ? event.data : es.url)));
                document.body.appendChild(div);
            }
        };
        var errlistener = function(event) {
            es.close();
        }
        es.addEventListener("open", listener);
        es.addEventListener("message", listener);
        es.addEventListener("error", errlistener);
    </script>
</head>

<body>
</body>

</html>
4

3 回答 3

6

将卡住的数据返回到浏览器的最佳方法是使用 Web Sockets 让客户端打开一个到文件阅读器的套接字,然后您可以毫无问题地将数据分块到浏览器。

然后,一旦完成,您就可以关闭套接字。

一个很好的网络套接字教程 http://www.phpbuilder.com/articles/application-architecture/optimization/creating-real-time-applications-with-php-and-websockets.html

使用这种方法,如果你想实现验证,那么服务器不仅可以发送块,它还会根据 javascript 的请求发送块

所以你的客户可以说我需要块 5 并且你的服务器实现了类似的东西

$requestedChunk = 5; // this would be set by the javascript sending the request
$chunkSize = 256; // this would be your chunk size;

$readPossition = $requestedChunk * $chunkSize;

链接不再有效,所以这里是一个建立在 Ratchet 上的链接:https ://blog.samuelattard.com/the-tutorial-for-php-websockets-that-i-wish-had-existed/

于 2015-12-01T11:36:49.950 回答
0

我有一个类似的问题。事件流在使用 Apache 2.0 处理程序的服务器上按预期工作(返回块),但在使用 FastCGI 的服务器上却没有(批量返回)。我认为 FastCGI 中的某些东西是罪魁祸首,因此试图通过切换到 CGI 来解决问题。现在事件流按预期工作。

无论是使用 CGI 还是 FastCGI,服务器 API 都显示为 CGI/FastCGI,所以我假设它为您工作的服务器正在运行 CGI,而它不为您工作的服务器正在运行 FastCGI。尝试将非工作服务器更改为 CGI。

至于为什么它在 FastCGI 中不起作用,我不完全确定,但除非它是必要的要求并且 CGI 是不可能的,否则上述解决方案应该可以工作。

于 2015-11-27T16:57:39.467 回答
0

许多事情可以防止分块响应,例如但不限于;

  • Web 服务器上的代理或任何其他缓冲机制
  • 当 php.ini 中的“输出缓冲”为“打开”时(您应该明确将其设置为关闭)
  • 在 Web 服务器上启用 gzip 时

你应该首先检查这些。

于 2015-12-04T15:16:36.030 回答