22

Hi Please View Below Code :

<?php
ob_start();

echo "Start ...<br />\n";
for( $i = 0 ; $i < 10 ; $i++ )
{
    echo "$i<br />\n";
    ob_flush(); 
    flush();
    sleep(1);
}
echo "End ...<br />\n";
?>

It's Incorrect ? i'm tested it but my output show when script is done, have any solution ?

DJafari
  • 12,955
  • 8
  • 43
  • 65

12 Answers12

14

Hey man I was also got stuck in this problem and finally got the correct solution here it is for you

you have to add content type for your page you can do that by two ways 1. using html tag

<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">

Ex.

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Wp Migration</title>
</head>
<body>
<?php 
for($i=0;$i<70;$i++)
{
echo 'printing...<br>';
ob_flush();
flush();
sleep(3);
}
?>
</body>
</html>
  1. using php header function

    <?php header( 'Content-type: text/html; charset=utf-8' ); ?>

Ex.

<?php 
header( 'Content-type: text/html; charset=utf-8' );
for($i=0;$i<70;$i++)
{
echo 'printing...<br>';
ob_flush();
flush();
sleep(3);
}
?>

All the best

Rahul Shinde
  • 853
  • 9
  • 7
  • So it seems the browser gets the content with or without the content type line, however if you don't have it, the browser simply changes the loading indicator to "receiving data" without displaying anything. Once you send the content type first, sending any amount of data and flushing will work =))) I confirmed the same behavior exists when running on Apache and IIS (regardless of output buffering settings) so it's definitely due to the browser's own requirements. – Yavor Feb 19 '15 at 20:41
  • ob_flush(): Failed to flush buffer. No buffer to flush – Anton Duzenko Jul 07 '23 at 16:38
6

Some browsers need to receive at least 256 characters before they start to render. Have you already tried to stuff more output like:

echo str_repeat('&nbsp;', 50) . "$i<br />\n";

EDIT:

Under Apache/2.2.11 (Win32) DAV/2 mod_ssl/2.2.11 OpenSSL/0.9.8i PHP/5.2.9 I was able to reproduce the problem of the OP by setting

zlib.output_compression = On

Turning it off again by

zlib.output_compression = Off

made the script work as wanted.

Jürgen Thelen
  • 12,745
  • 7
  • 52
  • 71
4

Try removing the call to ob_start() on your first line : there is no need for you to enable output buffering -- and it probably causes troubles, here.


I've tested your code :

  • If ob_start() is called on the first line, I only see the output when the script finishes, after 10 seconds
  • If I remove that call to ob_start(), then, I see one line of output every second, as soon as it's displayed on the standard output.
Pascal MARTIN
  • 395,085
  • 80
  • 655
  • 663
  • If you remove `ob_start()`, then you should remove all the `ob_` calls as well. – Jason McCreary Apr 24 '11 at 14:10
  • Not sure about that *(If I remove the call to `ob_flush()`, I only see the output after the 10 seconds delay)* ; maybe it's because `output_buffering` is enabled in my php.ini, though ;;; see what's written on http://fr.php.net/flush too – Pascal MARTIN Apr 24 '11 at 14:13
  • are you sure ? tested in browser ? – DJafari Apr 24 '11 at 14:14
  • is possible give me link to view result ? – DJafari Apr 24 '11 at 14:19
  • I tested on my local computer *(and the Apache I've got on it is not accessible from the Internet)* ;; anyway, I just copy-pasted your code, and commented the `ob_start()` call. – Pascal MARTIN Apr 24 '11 at 14:22
  • I Tested it on local ( WAMP5 ) and Apache Web Server, both return output after 10 second ! i even tested this code on 2 server, but problem still exist . – DJafari Apr 24 '11 at 14:25
2

Using Chrome, I found out that many more bytes are required to by-pass the browser's buffer. In my case 4096 bytes was fine:

echo str_repeat(' ', 4096);

Also, adding some HTML element at the beginning also seemed to be mandatory:

echo $content . '<br />';
1

I've discovered that this was due to Apache's gzip compression being in use for my case.

To turn gzip off for the 'flushing' script only, I created a new .htaccess file in the directory where the continuous output script resides, with the following:

<IfModule mod_env.c>
    SetEnv no-gzip 1
</IfModule>

Flushing is working as expected again.

Josh Harrison
  • 5,927
  • 1
  • 30
  • 44
1

For people using FCGI / fast cgi.

FcgidOutputBufferSize 0
Alex S
  • 147
  • 2
  • 3
1

On my system it appears that FF4 needs more than 256 bytes to start rendering what is arriving from the server side, then i resolved with this at the beginning:

while (@ob_end_flush());
echo(str_repeat(' ',1024));
// ...etc...
DarthJDG
  • 16,511
  • 11
  • 49
  • 56
Ulix
  • 11
  • 1
0

One sneaky issue with IE8 and flush(); is that if you're "flushing" out rows in a table. IE will only display tables when they're complete. This was my issue, and changing containers from table rows to divs solved the problem.

0

You need to add a .htaccess file to disable gzip output

<IfModule mod_env.c>
    SetEnv no-gzip 1
</IfModule>
Mi-Creativity
  • 9,554
  • 10
  • 38
  • 47
Dylan B
  • 796
  • 8
  • 16
0

I am using laravel framework and buffering did not work but. This is solution :

header( 'Content-type: text/html; charset=utf-8' );
ob_start();

ob_end_flush();
ob_flush();
flush();
for($i = 1;$i<= 5;$i++){
    echo $i;
    ob_flush();
    flush();

    sleep(3);
}

You have to use first ob_end_flush();

Ahmet Uğur
  • 462
  • 6
  • 12
0

This flow works with Laravel too

ob_implicit_flush(true);
echo "Processing ... "; // Or give out JSON output
ob_flush();
sleep(5); //A time-consuming synchronous process (SMTP mail, maybe?)
echo "Done";
Deepak Thomas
  • 3,355
  • 4
  • 37
  • 39
0

It is correct. Works fine for me from CLI running PHP 5.3.3. If it's not working for you, your PHP install may have output buffering disabled.

I would also suggest putting ob_end_flush() at the end of your script to close the output buffer.

Jason McCreary
  • 71,546
  • 23
  • 135
  • 174