先看看这个函数:
fopen(打开文件)
相关函数
open,fclose
表头文件
#include<stdio.h>
定义函数
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<stdio.h>main(){FILE * fp;fp=fopen("noexist","a+");if(fp= =NULL) return;fclose(fp);}
fread(从文件流读取数据)
相关函数
fopen,fwrite,fseek,fscanf
表头文件
#include<stdio.h>
定义函数
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<stdio.h>#define nmemb 3struct 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;i<nmemb;i++)printf("name[%d]=%-20s:size[%d]=%d/n",i,s[i].name,i,s[i].size);}
执行
name[0]=Linux! size[0]=6name[1]=FreeBSD! size[1]=8name[2]=Windows2000 size[2]=11
fwrite(将数据写至文件流)
相关函数
fopen,fread,fseek,fscanf
表头文件
#include<stdio.h>
定义函数
size_t fwrite(const void * ptr,size_t size,size_t nmemb,FILE * stream);
函数说明
fwrite()用来将数据写入文件流中。参数stream为已打开的文件指针,参数ptr 指向欲写入的数据地址,总共写入的字符数以参数size*nmemb来决定。Fwrite()会返回实际写入的nmemb数目。
返回值
返回实际写入的nmemb数目。
范例
#include<stdio.h>#define set_s (x,y) {strcoy(s[x].name,y);s[x].size=strlen(y);}#define nmemb 3struct 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);}
而文件的定位很少有帖子写到。比如说把一个源文件拷贝成另外的10个目标文件,则每次拷贝完一个目标文件后都要对源文件重新定位。从文件开头开始重新读取文件,然后再写文件。
ret = fseek(sourcefp , 0 ,SEEK_SET );
if (ret <0)
{
perror("pointer fail/n");
exit(0);
}
还有另外一个问题,就是如何得到文件结束的标志。如何判断文件已经读完了,已经读到文件尾部。我试验了下,这个还是最精确的。
//read the source file
while(!feof(sourcefp))
{
buf_len = fread(data,sizeof(unsigned char),sizeof(data),sourcefp);
fwrite(data,1,buf_len,desfp);
}
feof来确定文件已经读到尾部。跳出循环。
一下是我写的一个小程序,把1.rmvb复制成2000份,分别复制成2.rmvb,3.rmvb、、、2000.rmvb。完整程序如下:(见笑)
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/types.h>
#include <string.h>
#define FILENUM 2000
#define FILENAME ("1.rmvb")
int main(int argc , char * argv[])
{
int i;//count
FILE * sourcefp;//resource file
FILE * desfp;//destination file
char tempfilename[20];// temp file name
char data[10240];
int count=2;//file count
int ret;//ce shi used
int buf_len;//buf length
//open the resource file
sourcefp = fopen(FILENAME , "a+");
if (sourcefp == NULL)
{
perror("open file fail/n");
//exit(0);
}
for(i=2 ;i<=FILENUM ;i++)
{
//filename
memset(tempfilename , 0 ,sizeof(tempfilename));
bzero(tempfilename , sizeof(tempfilename));
sprintf(tempfilename , "%d.rmvb" ,count);
count++;//count add
//open destination file
desfp = fopen(tempfilename , "ab+");
if (desfp == NULL )
{
printf("open file %d.rmvb fail ",count);
continue;//if fail continue;
}
printf("第%d个文件开始复制.../n",count);
//point the pointer to the file head
ret = fseek(sourcefp , 0 ,SEEK_SET );
if (ret <0)
{
perror("pointer fail/n");
exit(0);
}
//read the source file
while(!feof(sourcefp))
{
buf_len = fread(data,sizeof(unsigned char),sizeof(data),sourcefp);
fwrite(data,1,buf_len,desfp);
}
//close temp file point
printf("第%d个文件复制完毕/n",count);
fclose(desfp);
printf("%s/n" ,tempfilename);
}
fclose(sourcefp);
printf("%s" , FILENAME);
return 0;
}
完成。在linux下编译通过