phpキャッシュobシリーズ関数の解析(再掲)
転載元アドレス:
http://hi.baidu.com/imdao/blog/item/723e7e8b589b5e11c8fc7ada.html
for ($i=10; $i>0 ; $i--)
{
echo $i;
flash();
sleep(1);
}
?>
フォロー phpマニュアルによると
この関数は、これまでのプログラムのすべての出力をユーザーのブラウザに送信します。
上記のコードは $i を毎秒出力する必要があります。しかし、実際には必ずしもそうであるとは限りません。 10 秒間待機した後、すべての出力が同時に表示される可能性があります。
わかりました。このコードを
ob_end_clean();//
の部分を変更します ($i=10; $ i>0; $ i--)
{
echo $i;
flash();
sleep(1);
}
?>
ねえ、追加した後この文 ob_end_clean(); 、大丈夫です。実際、ob_end_clean() を ob_end_flush() に置き換えても問題ありません。
また変更させていただきます。
for ($i=10; $i>0; $i--)
{
echo $i;
ob_flush();//
flash();
sleep(1);
}
?>
の部分を修正して実行してみると、$i も毎秒出力されているのがわかりますか?これはなぜでしょうか?
心配しないで、php.ini を見てみましょう。
php.ini を開き、output_buffering を検索すると、output_buffering = 4096 のような設定が表示されます。 Output_buffering という名前と同じように、この設定の機能は出力をバッファリングすることです。バッファ サイズは 4096 バイトです。
最初のコードで、出力が期待どおりにならないのは、まさに次のような理由によるものです。 this Output_buffering それらの出力をバッファリングします。出力は、4096 バイトに達するか、スクリプトが終了するまで送信されません。
2 番目のコードの ob_end_clean() と ob_end_flush() の機能は、バッファリングを終了することです。この方法では、送信前に 4096 バイトのバッファができるまで待つ必要はありません。
3 番目のコードでは、ob_flush() が使用されます。その機能はバッファリングされたデータを送信することですが、バッファリングは終了しないため、各フラッシュ() の前に使用する必要があります。
ob_end_clean()、ob_end_flush()、および ob_flush() を使用したくない場合は、php.ini の Output_buffering を十分小さい値 (たとえば、0 に設定) に設定する必要があります。 ini_set("output_buffering","0") を使用してスクリプト内で設定する予定がある場合は、中止してください。このメソッドは機能しません。スクリプトの開始時にバッファリング設定が読み込まれ、その後バッファリングが開始されるためです。
ob_flush() はバッファリングされたデータを送信するのに、なぜフラッシュ() を使用する必要があるのかと疑問に思われるかもしれません。次のコードを使用するだけではだめでしょうか? ?
for ($i=10; $i>0; $i--)
{
echo $i;
ob_flush();
sleep(1);
}
?>
ob_flush() と flash() の違いに注意してください。前者はPHPのバッファからデータを解放することであり、後者はバッファにないデータまたは解放されたデータをブラウザに送信することです。したがって、バッファが存在する場合は、ob_flush() と flash() を同時に使用する必要があります。
ここではflush()が必須ですか?いいえ、データが出力されるとすぐにブラウザーに送信される別の方法があります。次の 2 つのコードでは、flush() を使用する必要はありません。 (output_buffering を 0 に設定すると、ob_flush() と ob_end_clean() も必要なくなります)
ob_implicit_flush(true);
for ($i= 10; $i>0; $i--)
{
echo $i;
ob_flush();
sleep(1);
}
?>
ob_end_clean();
ob_implicit_flush(true);
for ($i=10; $i>0; $i--)
{
echo $ i;
sleep(1);
}
?>
上記の ob_implicit_flush(true) に注意してください。この関数は、出力があるたびに出力を強制的にフラッシュします。出力はブラウザに送信されます。この方法では、各出力 (エコー) の後にフラッシュ() を使用してブラウザに送信する必要はありません。
上記の問題は、一部のブラウザでは当てはまらない場合があります。ブラウザにも独自のルールがあるためです。テストにはFirefox1.5、IE6、opera8.5を使用しました。このうち Opera は HTML タグに遭遇しないとスクリプトが終了しない限り出力しないルールとなっているため、正常に出力することができません。 FireFoxとIEは比較的普通です。
最後に、PuTTYshell によって書かれた非常に興味深いコードが添付されています。スクリプト サイクルでは、各出力が前の出力を上書きします。
次のコードは、Firefox でのみ使用できます。他のブラウザでは、multipart/x-mixed-replace の Content-Type がサポートされません。
header('Content-type: multipart/x-mixed-replace;boundary=endofsection');
print "n--endofsectionn";
$pmt = array("-", "\" , "|", "/" );
for( $i = 0; $i <10; $i ++ ){
sleep(1);
print "Content-type: text /plainnn";
print "Part $it".$pmt[$i % 4];
print "--endofsectionn";
ob_flush();
flash();
}
print "Content-type: text/plainnn";
print "The endn";
print "--endofsection--n";
?>