本文环境: OS:Mac OS X 10.8.4 PHP:5.3.15
PHP的官方手册中,函数feof()下面的讨论不少,对此做了一些相关的测试。
1 <?php 2 print <<<EOF 3 <!DOCTYPE html> 4 <html> 5 <head> 6 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> 7 <title>测试PHP中的feof()函数效果</title> 8 </head> 9 <body> 10 <div> 11 EOF; 12 13 function bool2str($bool) { 14 if ($bool == TRUE) { 15 return "TRUE"; 16 } else { 17 return "FALSE"; 18 } 19 } 20 21 /* 22 * 请随便创建一个文件。 23 * 比如:本测试中,在脚本文件的相同路径下创建了一个文本文件, 24 * 文件内容为“abcdefg”,文件名为“7bytesfile”。 25 */ 26 $filename = './7bytesfile'; 27 $handle = fopen($filename, 'r'); 28 if (!$handle) { 29 die("文件打开失败"); 30 } 31 32 for($i = 0; $i <= filesize($filename); $i++) { 33 fseek($handle, $i); 34 echo "文件位置" . ftell($handle) . ":<br />\n"; 35 echo "执行fseek,尚未执行读取操作之前,feof结果:" . bool2str(feof($handle)) . "<br />\n"; 36 echo "当前位置字符:" . fgetc($handle) . "<br />\n"; 37 echo "执行文件读取操作之后,feof结果:" . bool2str(feof($handle)) . "<hr />\n"; 38 } 39 /* 40 * 通过上面一段代码可以观察到, 41 * 随着循环的执行,文件指针从文件头一直移动到文件末尾。 42 * 但是当完成了字符“g”的读取输出,文件指针继续向后移动,这是feof()依然返回False。 43 * 只有当执行了一次fgetc()操作之后,才返回true,表示到达文件末尾。 44 */ 45 46 echo "ftell()结果:". ftell($handle). "<hr />\n"; 47 //输出一下,很郁闷的发现文件指针的位置还是7。+_+ 48 49 fseek($handle, 4); 50 echo "文件位置" . ftell($handle) . ":<br />\n"; 51 echo "执行fseek,尚未执行读取操作之前,feof结果:" . bool2str(feof($handle)) . "<br />\n"; 52 echo "当前位置字符:" . fgetc($handle) . "<br />\n"; 53 echo "执行文件读取操作之后,feof结果:" . bool2str(feof($handle)) . "<hr />\n"; 54 55 fseek($handle, 7); 56 echo "文件位置" . ftell($handle) . ":<br />\n"; 57 echo "执行fseek,尚未执行读取操作之前,feof结果:" . bool2str(feof($handle)) . "<br />\n"; 58 echo "当前位置字符:" . fgetc($handle) . "<br />\n"; 59 echo "执行文件读取操作之后,feof结果:" . bool2str(feof($handle)) . "<hr />\n"; 60 fclose($handle); 61 //再次移动文件指针,效果依旧。 62 //再用另外一段代码测试一下: 63 64 $handle = fopen($filename, 'r'); 65 if (!$handle) { 66 die("文件打开失败"); 67 } 68 while (!feof($handle)) { 69 $char = fgetc($handle); 70 if ($char === FALSE) { 71 echo 'FALSE'; 72 } else { 73 echo $char; 74 } 75 } 76 fclose($handle); 77 //依然是输出了字符g之后,再次执行读取操作,才终止循环。 78 79 print <<<EOF 80 </div> 81 </body> 82 </html> 83 EOF; 84 ?>
针对这种情况的猜测是,在PHP中,feof()的实现方式并非直接检查文件指针相对于文件的位置,而是根据某个标识返回结果。每次fseek()之后都会都会把这个标识设置为“False”,只有当执行一次文件内容读取操作之后,才会根据文件读取的结果对标识进行设置。
根据这种猜测,可以使用两种代码逻辑。
一个方法是不做feof()检测,直接检测内容读取函数(比如fgetc()、fgets())的执行结果。
1 while (($content = fgets($fileHandle)) !==FALSE) { 2 //文件内容处理…… 3 }
这种处理办法,利用了PHP被诟病的函数返回方式,所以得用“===”或“!==”进行检测,不能把代码简化成:
while ($content = fgets($fileHandle)) {}
另外一个方法是先进行一次文件读取,然后再进入feof()循环:
1 $content = fgets($fileHandle); 2 while (!feof($fileHandle)) { 3 //处理文件内容…… 4 $content = fgets($fileHandle); 5 }
经过测试,似乎前一种方法效率会高一些。