我有一个ob_start()
和一个相应的ob_flush()
。我想刷新一部分数据并继续执行其余的数据。使用ob_flush()
没有帮助。如果可能的话,休息需要在不显示浏览器加载的情况下发生
修改
我不想使用ajax
答案 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
。尽管如此,所有人都关闭了请求。