本文环境:
OS:Mac OS X 10.8.4
PHP:5.3.15

  PHP的官方手册中,函数feof()下面的讨论不少,对此做了一些相关的测试。

 <?php
print <<<EOF
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>测试PHP中的feof()函数效果</title>
</head>
<body>
<div>
EOF; function bool2str($bool) {
if ($bool == TRUE) {
return "TRUE";
} else {
return "FALSE";
}
} /*
* 请随便创建一个文件。
* 比如:本测试中,在脚本文件的相同路径下创建了一个文本文件,
* 文件内容为“abcdefg”,文件名为“7bytesfile”。
*/
$filename = './7bytesfile';
$handle = fopen($filename, 'r');
if (!$handle) {
die("文件打开失败");
} for($i = 0; $i <= filesize($filename); $i++) {
fseek($handle, $i);
echo "文件位置" . ftell($handle) . ":<br />\n";
echo "执行fseek,尚未执行读取操作之前,feof结果:" . bool2str(feof($handle)) . "<br />\n";
echo "当前位置字符:" . fgetc($handle) . "<br />\n";
echo "执行文件读取操作之后,feof结果:" . bool2str(feof($handle)) . "<hr />\n";
}
/*
* 通过上面一段代码可以观察到,
* 随着循环的执行,文件指针从文件头一直移动到文件末尾。
* 但是当完成了字符“g”的读取输出,文件指针继续向后移动,这是feof()依然返回False。
* 只有当执行了一次fgetc()操作之后,才返回true,表示到达文件末尾。
*/ echo "ftell()结果:". ftell($handle). "<hr />\n";
//输出一下,很郁闷的发现文件指针的位置还是7。+_+ fseek($handle, 4);
echo "文件位置" . ftell($handle) . ":<br />\n";
echo "执行fseek,尚未执行读取操作之前,feof结果:" . bool2str(feof($handle)) . "<br />\n";
echo "当前位置字符:" . fgetc($handle) . "<br />\n";
echo "执行文件读取操作之后,feof结果:" . bool2str(feof($handle)) . "<hr />\n"; fseek($handle, 7);
echo "文件位置" . ftell($handle) . ":<br />\n";
echo "执行fseek,尚未执行读取操作之前,feof结果:" . bool2str(feof($handle)) . "<br />\n";
echo "当前位置字符:" . fgetc($handle) . "<br />\n";
echo "执行文件读取操作之后,feof结果:" . bool2str(feof($handle)) . "<hr />\n";
fclose($handle);
//再次移动文件指针,效果依旧。
//再用另外一段代码测试一下: $handle = fopen($filename, 'r');
if (!$handle) {
die("文件打开失败");
}
while (!feof($handle)) {
$char = fgetc($handle);
if ($char === FALSE) {
echo 'FALSE';
} else {
echo $char;
}
}
fclose($handle);
//依然是输出了字符g之后,再次执行读取操作,才终止循环。 print <<<EOF
</div>
</body>
</html>
EOF;
?>

  针对这种情况的猜测是,在PHP中,feof()的实现方式并非直接检查文件指针相对于文件的位置,而是根据某个标识返回结果。每次fseek()之后都会都会把这个标识设置为“False”,只有当执行一次文件内容读取操作之后,才会根据文件读取的结果对标识进行设置。

  根据这种猜测,可以使用两种代码逻辑。

  一个方法是不做feof()检测,直接检测内容读取函数(比如fgetc()、fgets())的执行结果。

 while (($content = fgets($fileHandle)) !==FALSE) {
//文件内容处理……
}

  这种处理办法,利用了PHP被诟病的函数返回方式,所以得用“===”或“!==”进行检测,不能把代码简化成:

while ($content = fgets($fileHandle)) {}

  另外一个方法是先进行一次文件读取,然后再进入feof()循环:

 $content = fgets($fileHandle);
while (!feof($fileHandle)) {
//处理文件内容……
$content = fgets($fileHandle);
}

  经过测试,似乎前一种方法效率会高一些。

最新文章

  1. Guidance of Set up FTP Server
  2. windows下多字节和宽字节转换
  3. 编译原理(简单自动词法分析器LEX)
  4. 哇 真的是一个好插件!!!Sublime Text编辑文件后快速刷新浏览器
  5. d.BIO连接器与NIO连接器的对比之二
  6. OSI安全体系结构
  7. bzoj 2716 天使玩偶(CDQ分治,BIT)
  8. eoe推荐的优秀博客
  9. C++程序在debug模式下遇到Run-Time Check Failure #0 - The value of ESP was not properly saved across a function call问题。
  10. javascript EcmaScript5 新增对象之Object.freeze
  11. verilog中读取文件中的字符串_modelsim高级仿真
  12. Javascript 正确用法 二
  13. python笔记13-文件读写
  14. isinstance, type, issubclass
  15. 再读c++primer plus 002
  16. MySQL优化系列
  17. Ckeditor的JS的加载和取值和赋值方法
  18. iis配置asp.net常见验证失败问题解决方案
  19. IOS - IPhone或IPAD,如何恢复出厂操作系统?
  20. Nginx - 日志格式及输出

热门文章

  1. scrapy爬虫学习系列一:scrapy爬虫环境的准备
  2. RDIFramework.NET ━ .NET快速信息化系统开发框架 V3.2-&gt;Web版本“产品管理”事例编辑界面新增KindEditor复文本编辑控件
  3. ASP.NET Core使用Jaeger实现分布式追踪
  4. Java复制、移动和删除文件
  5. MySQL数据库性能优化(享学课堂听课笔记)
  6. 解决echarts饼图不显示数据为0的数据
  7. Java多线程编程实战读书笔记(一)
  8. sql server去掉某个字段前后空格问题
  9. Dynamics 365中配置和使用文件夹级别的跟踪(folder-level tracking)
  10. iOS----------APP怎样做更安全