我一直觉得二进制文件读写是个很容易的事,所以一直没在意,最近在写一个http客户端,实现文件下载的时候,发现总有问题,后来才发现是忘记写文件用二进制方式,惭愧的很啊。 然后,就在网上搜索了一下,发现通过C语言实现二进制文件读写的资料居然出奇的少,这让我很愤怒,因为虽然这东西很简单,但是对于初学者,往往会需要花很长的时间去弄,一旦明白,又发现花的时间很不值得,罢了,这里通过一个文件拷贝的例子来讲讲二进制文件的读写吧。 先介绍函数,我们一共要用到三个函数,fopen,fread,fwrite。二进制读写的顺序是用fopen以二进制方式打开读写文件,然后使用fread和fwrite两个函数将数据写入二进制文件中。下面我们看看一个拷贝程序的源码:
Copy.c: #include #include
#define MAXLEN 1024
int main(int argc, char *argv[]) { if( argc < 3 ) { printf("usage: %s %s\n", argv[0], "infile outfile"); exit(1); } FILE * outfile, *infile; outfile = fopen(argv[2], "wb" ); infile = fopen(argv[1], "rb"); unsigned char buf[MAXLEN]; if( outfile == NULL || infile == NULL ) { printf("%s, %s",argv[1],"not exit\n"); exit(1); } int rc; while( (rc = fread(buf,sizeof(unsigned char), MAXLEN,infile)) != 0 ) { fwrite( buf, sizeof( unsigned char ), rc, outfile ); } fclose(infile); fclose(outfile); system("PAUSE"); return 0; } |
现在来讲讲这个程序,这个程序的作用就是将文件1的内容直接拷贝到文件2中,注意fread的返回值,这个值需要在fwrite的时候将会用到。
后面是关于fopen,fread,fwrite三个函数的详细说明。 fopen(打开文件) 相关函数 | open,fclose | 表头文件 | #include | 定义函数 | FILE * fopen(const char * path,const char * mode); | 函数说明 | 参数path字符串包含欲打开的文件路径及文件名,参数mode字符串则代表着流形态。 mode有下列几种形态字符串: r 打开只读文件,该文件必须存在。 r+ 打开可读写的文件,该文件必须存在。 w 打开只写文件,若文件存在则文件长度清为0,即该文件内容会消失。若文件不存在则建立该文件。 w+ 打开可读写文件,若文件存在则文件长度清为零,即该文件内容会消失。若文件不存在则建立该文件。 a 以附加的方式打开只写文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾,即文件原先的内容会被保留。 a+ 以附加方式打开可读写的文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾后,即文件原先的内容会被保留。 上述的形态字符串都可以再加一个b字符,如rb、w+b或ab+等组合,加入b 字符用来告诉函数库打开的文件为二进制文件,而非纯文字文件。不过在POSIX系统,包含Linux都会忽略该字符。由fopen()所建立的新文件会具有S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH(0666)权限,此文件权限也会参考umask值。 | 返回值 | 文件顺利打开后,指向该流的文件指针就会被返回。若果文件打开失败则返回NULL,并把错误代码存在errno 中。 | 附加说明 | 一般而言,开文件后会作一些文件读取或写入的动作,若开文件失败,接下来的读写动作也无法顺利进行,所以在fopen()后请作错误判断及处理。 | 范例 | #include main() { FILE * fp; fp=fopen("noexist","a+"); if(fp= =NULL) return; fclose(fp); } |
fread(从文件流读取数据) 相关函数 | fopen,fwrite,fseek,fscanf | 表头文件 | #include | 定义函数 | size_t fread(void * ptr,size_t size,size_t nmemb,FILE * stream); | 函数说明 | fread()用来从文件流中读取数据。参数stream为已打开的文件指针,参数ptr 指向欲存放读取进来的数据空间,读取的字符数以参数size*nmemb来决定。Fread()会返回实际读取到的nmemb数目,如果此值比参数nmemb 来得小,则代表可能读到了文件尾或有错误发生,这时必须用feof()或ferror()来决定发生什么情况。 | 返回值 | 返回实际读取到的nmemb数目。 | 附加说明 |
| 范例 | #include #define nmemb 3 struct test { char name[20]; int size; }s[nmemb]; main() { FILE * stream; int i; stream = fopen("/tmp/fwrite","r"); fread(s,sizeof(struct test),nmemb,stream); fclose(stream); for(i=0;iprintf("name[%d]=%-20s:size[%d]=%d\n",i,s[i].name,i,s[i].size); } | 执行 | name[0]=Linux! size[0]=6 name[1]=FreeBSD! size[1]=8 name[2]=Windows2000 size[2]=11 |
fwrite(将数据写至文件流) 相关函数 | fopen,fread,fseek,fscanf | 表头文件 | #include | 定义函数 | size_t fwrite(const void * ptr,size_t size,size_t nmemb,FILE * stream); | 函数说明 | fwrite()用来将数据写入文件流中。参数stream为已打开的文件指针,参数ptr 指向欲写入的数据地址,总共写入的字符数以参数size*nmemb来决定。Fwrite()会返回实际写入的nmemb数目。 | 返回值 | 返回实际写入的nmemb数目。 | 范例 | #include #define set_s (x,y) {strcoy(s[x].name,y);s[x].size=strlen(y);} #define nmemb 3 struct test { char name[20]; int size; }s[nmemb]; main() { FILE * stream; set_s(0,"Linux!"); set_s(1,"FreeBSD!"); set_s(2,"Windows2000."); stream=fopen("/tmp/fwrite","w"); fwrite(s,sizeof(struct test),nmemb,stream); fclose(stream); } | 执行 | 参考fread()。 |
|
|