最近工作比较清闲,于是闲着没事翻着看《C缺陷与陷阱》,在第五章库函数这一章节的时候,提到了不能自由的同时读写文件。即不能fwrite写之后,立刻fread读取。
正好电脑上有安装好的ubuntu虚拟机,于是就像动手验证下。
具体代码如下:
#include <stdio.h> #include <string.h> int main(void) { FILE *fd; int ret; fd = fopen("test.txt","w"); if ( ret <0) perror("open error"); char str[]="hi hero'; ret = fwrite(str,sizeof(char),strlen(str),fd); if ( ret <0) perror("write error"); ret = fseek(fd,0L,SEEK_SET); if ( ret <0) perror("seek error"); char str2[20]=""; ret = fread(str2,sizeof(char),strlen(str),fd); if ( ret <=0) perror("read error"); printf("%s",str2); return 0; }
结果在fread的时候报错了,提示说bad file descrition,大致是说错误的文件描述符。于是gdb调试,打印出来的fd文件描述符也没错,很郁闷了。后来在fwrite之后添加了fclose函数,然后再次fopen,这样程序就可以正常运行了。估计可能的原因是我写的时候,内容实际是在内容中,而没有实际的同步到磁盘中,于是fread报错了。添加fclose之后,关闭文件会把内存中的动向同步至磁盘中,这样再次打开读取正常了。刚才回去又看了眼代码,发现代码中有个很明显的权限错误,fopen的时候指定了"w"只写权限,难怪fread失败了,呵呵。改过来之后,继续测试。
还是依然会报错,bad file descrition,于是把fseek的注释去掉,程序正常工作了!看来还是实践出真知,c语言中的确不能简单的写之后立刻进行读操作,必须加入一个fseek语句。#include <stdio.h> #include <string.h> int main(void) { FILE *fd; int ret; fd = fopen("test.txt","w+"); if ( ret <0) perror("open error"); char str[]="hi hero'; ret = fwrite(str,sizeof(char),strlen(str),fd); if ( ret <0) perror("write error"); /* ret = fseek(fd,0l,SEEK_SET); if ( ret <0) perror("seek error"); */ char str2[20]=""; ret = fread(str2,sizeof(char),strlen(str),fd); if ( ret <=0) perror("read error"); printf("%s",str2); return 0; }
而之前猜测的是不是内存内容未同步到磁盘的想法,验证了之后发现也不是这个的原因。
代码修改如下:
#include <stdio.h> #include <string.h> int main(void) { FILE *fd; int ret; fd = fopen("test.txt","w+"); if ( ret <0) perror("open error"); char str[]="hi hero'; ret = fwrite(str,sizeof(char),strlen(str),fd); if ( ret <0) perror("write error"); ret = fflush(fd); if ( ret <0) perror("fflush error"); char str2[20]=""; ret = fread(str2,sizeof(char),strlen(str),fd); if ( ret <=0) perror("read error"); printf("%s",str2); return 0; }
fread函数还是会报bad file descirption。大致就是这样样子了,具体为什么不能同时读写的原因书上说是为了保持与以前不能同时读写文件的程序兼容。有更了解的朋友可以留言讨论下。呵呵。