fgetc, EOF 及其它
发表于:2007-05-26来源:作者:点击数:
标签:
在另一个贴子中,我与一些朋友对getc展开了一些讨论.由于觉得楼主最终未能明白 我的意思,所以我把我个人的看法总结出来,写在这里.我不太擅长说明,但已经尽力了. 任何人转本贴,请务必把本人的名字写在显眼的位置.8) 约定编译器为gcc2/x86: 所以char,unsignedch
在另一个贴子中,我与一些朋友对 getc 展开了一些讨论. 由于觉得楼主最终未能明白
我的意思,所以我把我个人的看法总结出来,写在这里.我不太擅长说明,但已经尽力了.
任何人转本贴, 请务必把本人的名字写在显眼的位置. 8)
约定编译器为 g
clearcase/" target="_blank" >cc2/x86:
所以 char, unsigned char 为 8 位, int 为 32 位
请参考 http://bbs.chinaunix.net/forum/23/20031223/229236.html
(1) 字节的读取
在正常的情况下, getc 以 unsigned char 的方式读取文件流, 扩张为一个整数,并返
回. 换言之, getc 从文件流中取一个字节, 并加上24个零,成为一个小于256的整数,
然后返回.
int c;
while ((c = fgetc (rfp))!= -1) // -1就是 EOF
fputc (c, wfp);
上面 fputc 中的 c 虽然是整数, 但在 fputc 将其写入文件流之前, 又把整数的高24位
去掉了, 因此 fgetc, putc 配合能够实现文件复制. 到目前为止, 把 c 定义为
char仍然是可行的, 但下面我们将看到,把 c 定义为 int 是为正确判段文件是否结束.
(2) 判断文件结束.
多数人认为文件中有一个EOF,用于表示文件的结尾. 但这个观点实际上是错误的,在文
件所包含的数据中,并没有什么文件结束符. 对getc 而言, 如果不能从文件中读取,
则返回一个整数 -1,这就是所谓的EOF. 返回 EOF 无非是出现了两种情况,一是文件已
经读完; 二是文件读取出错,反正是读不下去了.
请注意: 在正常读取的情况下, 返回的整数均小于256, 即0x0~0xFF. 而读不出返回的
是 0xFFFFFFFF. 但, 假如你用fputc把 0xFFFFFFFF 往文件里头写, 高24位被屏蔽,写入的将
是 0xFF. // lixforalpha 请注意这一点
(3) 0xFF 会使我们混淆吗?
不会, 前提是, 接收返回值的 c 要按原型定义为 int.
如果下一个读取的字符将为 0xFF, 则
int c;
c = fgetc (rfp); // c = 0x000000FF;
if (c != -1) // 当然不等, -1 是 0xFFFFFFFF
fputc (wfp); // 噢, OXFF 复制成功.
字符0xFF, 其本身并不是EOF.
(4) 将 c 定义 char
假定下一个读取的字符为 0xFF 则
char c;
c = fgetc (rfp); // fgetc(rfp)的值为 0x000000FF, 暗中降为字节, c = 0xFF
if (c != -1) // 字符与整数比较? c 被带符号(signed)扩展为0xFFFFFFFF, 喔噢,
条件成立,文件复制提前退出.
while ((c=fgetc(rfp))!=EOF) 中的判别条件成立, 文件复制结束! 意外中止.
(5) 将 c 定义为 unsigned char;
当读到文件末尾, 返回 EOF 也就是 -1 时,
unsigned char c;
c = fgetc (rfp); // fgetc (rfp)的值为EOF,即-1,即0xFFFFFFFF, 降格为字节, c=0xFF
if ( c!= -1) // c 被扩展为 0x000000FF, 永远不回等于 0xFFFFFFFF
所以这次虽然能正确复制 0xFF, 但却不能判断文件结束. 事实上,在 c 为 uchar 时,
c != -1 是永远成立的, 一个高
质量的编译器, 比如 gcc会在编译时指出这一点.
(6) 为何需要feof?
FILE *fp;
fp 指向一个很复杂的数据结构, feof 是通过这个结构中的标志来判断文件是否结束的.
如果文件用 fgetc 读取, 刚好把最后一个字符读出时, fp 中的EOF标志不会打开,这时
用feof判断,将会得到文件尚未结束的结论.
fgetc 返回 -1 时, 我们仍无法确信文件已经结束, 因为可能是读取错误! 这时我们
需要 feof 和 ferror.
【发表回复】【查看CU论坛原帖】【添加到收藏夹】【javascript:window.close()">关闭】
lixforalpha 回复于:2003-12-29 23:40:15
|
多谢,吐血奉献
太好了,终于明白了些!
偶低层的功力实在太差了!
话说回来,有些书讲得实在很马虎 :-(
|
蓝色键盘 回复于:2003-12-30 08:48:51
|
win_hate啊,早上来了就看见你在这里法帖子,写的很详细,原创精华,你没意见吧!
呵呵
|
win_hate 回复于:2003-12-30 09:07:32
|
[quote:2d24ecb05b="蓝色键盘"]win_hate啊,早上来了就看见你在这里法帖子,写的很详细,原创精华,你没意见吧!
呵呵[/quote:2d24ecb05b]
谢谢老大!
|
蓝色键盘 回复于:2003-12-30 09:08:47
|
兄弟们因该学习win_hate同志追根到底的精神,论坛有几个这样的人,一定能够带动大家的积极性
|
lenovo 回复于:2003-12-30 09:15:03
|
嘿嘿,偶也顶一下!
以后要多发文章哦!
|
蓝色键盘 回复于:2003-12-30 09:25:51
|
lenovo在这里的贡献和精神大家有目共睹,以后准备让你当接班人呢!
加油吧!
相互提高。。。。
|
lenovo 回复于:2003-12-30 09:38:20
|
[quote:00d77380d9="蓝色键盘"]lenovo在这里的贡献和精神大家有目共睹,以后准备让你当接班人呢!
加油吧!
相互提高。。。。[/quote:00d77380d9]
谢谢鼓励!
惭愧呀,老大!
|
fieryfox 回复于:2003-12-30 10:18:40
|
总结的不错,提个建议:尽量不要用-1代替EOF。EOF是与系统有关的,某些系统的EOF不是-1。
|
lixforalpha 回复于:2003-12-30 15:45:51
|
谨以此献不辞劳苦给予详细解释的win_hate、flw、dadaball、aero、蓝色键盘等众多网友
:em02::em02: [url=http://lixforalpha.533.net/i_want_to_do/wmjd.pdf]完美解答[/url] :em02::em02:
|
原文转自:http://www.ltesting.net
- 评论列表(网友评论仅供网友表达个人看法,并不表明本站同意其观点或证实其描述)
-