如何将数据刷新到浏览器但继续执行

时间:2012-05-14 07:28:48

标签: php output-buffering

我有一个ob_start()和一个相应的ob_flush()。我想刷新一部分数据并继续执行其余的数据。使用ob_flush()没有帮助。如果可能的话,休息需要在不显示浏览器加载的情况下发生

修改

我不想使用ajax

7 个答案:

答案 0 :(得分:16)

ob_flush写入缓冲区。换句话说,ob_flush告诉PHP给Apache(或nginx / lighttpd / whatever)输出,然后让PHP忘记它。一旦Apache拥有输出,它就可以随心所欲地完成任务。 (换句话说,在ob_flush之后,它是否会立即被写入浏览器而无法控制。

所以,简短回答:没有保证可以做到这一点。

只是一个猜测,你可能正在寻找AJAX。每当人们在你正在进行页面内容加载时试图操纵时,AJAX几乎总是正确的路径。

如果您想在后台继续执行任务,可以使用详细here ignore_user_abort,但这通常不是最佳方法。你基本上失去了对该线程的控制权,在我看来,Web服务器线程不是重处理所属的地方。

我会尝试将其从面向网络的东西中提取出来。这可能意味着一个cron条目或者只是从PHP内部生成一个后台进程(虽然从脚本执行内部开始的进程不会随着脚本而死,并且脚本不会等到它在死亡之前完成)。

如果你确实走了那条路,那就意味着你甚至可以在必要时制作某种状态系统。然后,您可以监视执行情况并为用户定期更新进度。 (从技术上讲,你也可以创建一个带有ignore_user_abort ed脚本的状态系统,但它对我来说似乎并不干净。)

答案 1 :(得分:16)

我过去做过这个,这就是我解决它的方法:

ob_start();

/*
 * Generate your output here
 */ 

// Ignore connection-closing by the client/user
ignore_user_abort(true);

// Set your timelimit to a length long enough for your script to run, 
// but not so long it will bog down your server in case multiple versions run 
// or this script get's in an endless loop.
if ( 
     !ini_get('safe_mode') 
     && strpos(ini_get('disable_functions'), 'set_time_limit') === FALSE 
){
    set_time_limit(60);
}

// Get your output and send it to the client
$content = ob_get_contents();         // Get the content of the output buffer
ob_end_clean();                      // Close current output buffer
$len = strlen($content);             // Get the length
header('Connection: close');         // Tell the client to close connection
header("Content-Length: $len");     // Close connection after $len characters
echo $content;                       // Output content
flush();                             // Force php-output-cache to flush to browser.
                                     // See caveats below.

// Optional: kill all other output buffering
while (ob_get_level() > 0) {
    ob_end_clean();
}

正如我之前在几条评论中所说,你应该注意抓取你的内容,因为这会改变你的内容的长度,但不会改变它的标题。它还可以缓冲您的输出,因此它不会立即发送给客户端 您可以尝试让apache知道不要使用apache_setenv('no-gzip', '1'); gzip您的内容。但是,如果您使用重写规则转到您的页面,这将无法工作,从那时起它也将修改这些环境变量。至少,它是为我做的。

manual

中查看有关向用户展示内容的更多提示

答案 2 :(得分:4)

这是我的功能

function bg_process($fn, $arr) {
    $call = function($fn, $arr){
        header('Connection: close');
        header('Content-length: '.ob_get_length());
        ob_flush();
        flush();
        call_user_func_array($fn, $arr);
        };
    register_shutdown_function($call, $fn, $arr);
    }
在php关闭连接之后,

将函数包装到最后。当然浏览器会停止缓冲。

function test() {
    while (true) {
        echo 'this text will never seen by user';
        }
    }

这是如何调用函数

bg_process('test'); 

第一个参数是callable, 第二个参数是一个要传递给带有索引数组的“test”函数的数组

注意:我不在脚本的开头使用ob_start()

答案 3 :(得分:2)

我有一篇文章解释如何在我的博客上使用apache / mod_php来实现:http://codehackit.blogspot.com/2011/07/how-to-kill-http-connection-and.html希望这会有所帮助,欢呼

答案 4 :(得分:0)

fastcgi_finish_request

  

此函数将所有响应数据刷新到客户端并完成请求。这允许执行耗时的任务而不会断开与客户端的连接。

不适用于Apache。(PHP 5> = 5.3.3,PHP 7)

答案 5 :(得分:0)

如果您使用的是PHP-FPM:

ignore_user_abort(true);
fastcgi_finish_request();

以上两个功能是ignore_user_abort防止错误并且fastcgi_finish_request关闭客户端连接的关键因素。

答案 6 :(得分:-1)

使用:

header("Content-Length: $len");

..其中$len是要刷新到客户端的数据的长度。

我没有背景知道这可以在何时何地发挥作用,但我尝试了几个浏览器,并立即返回:

<?PHP 
    header("Content-length:5");
    echo "this is more than 5";
    sleep(5);
?>

编辑:Chrome,IE和Opera显示this,而FireFox显示this is more than 5。尽管如此,所有人都关闭了请求。