总结open与fopen的区别
1. 来源
从来源的角度看,两者能很好的区分开,这也是两者最显而易见的区别:
open
是UNIX系统调用函数(包括LINUX等),返回的是文件描述符(File Descriptor),它是文件在文件描述符表里的索引。fopen
是ANSIC标准中的C语言库函数,在不同的系统中应该调用不同的内核api。返回的是一个指向文件结构的指针。PS:从来源来看,两者是有千丝万缕的联系的,毕竟C语言的库函数还是需要调用系统API实现的。
2. 移植性
这一点从上面的来源就可以推断出来,`fopen`是C标准函数,因此拥有良好的移植性;而`open`是UNIX系统调用,移植性有限。如windows下相似的功能使用API函数`CreateFile`。
3. 适用范围
open
返回文件描述符,而文件描述符是UNIX系统下的一个重要概念,UNIX下的一切设备都是以文件的形式操作。如网络套接字、硬件设备等。当然包括操作普通正规文件(Regular File)。- fopen是用来操纵普通正规文件(Regular File)的。
4. 文件IO层次
如果从文件IO的角度来看,前者属于低级IO函数,后者属于高级IO函数。低级和高级的简单区分标准是:谁离系统内核更近。低级文件IO运行在内核态,高级文件IO运行在用户态。
5. 缓冲
- 缓冲文件系统
缓冲文件系统的特点是:在内存开辟一个“缓冲区”,为程序中的每一个文件使用;当执行读文件的操作时,从磁盘文件将数据先读入内存“缓冲区”,装满后再从内存“缓冲区”依此读出需要的数据。执行写文件的操作时,先将数据写入内存“缓冲区”,待内存“缓冲区”装满后再写入文件。由此可以看出,内存“缓冲区”的大小,影响着实际操作外存的次数,内存“缓冲区”越大,则操作外存的次数就少,执行速度就快、效率高。一般来说,文件“缓冲区”的大小随机器 而定。fopen, fclose, fread, fwrite, fgetc, fgets, fputc, fputs, freopen, fseek, ftell, rewind
等。 - 非缓冲文件系统
缓冲文件系统是借助文件结构体指针来对文件进行管理,通过文件指针来对文件进行访问,既可以读写字符、字符串、格式化数据,也可以读写二进制数据。非缓冲文件系统依赖于操作系统,通过操作系统的功能对文件进行读写,是系统级的输入输出,它不设文件结构体指针,只能读写二进制文件,但效率高、速度快,由于ANSI标准不再包括非缓冲文件系统,因此建议大家最好不要选择它。open, close, read, write, getc, getchar, putc, putchar
等。
一句话总结一下,就是open
无缓冲,fopen
有缓冲。前者与read
, write
等配合使用, 后者与fread
,fwrite
等配合使用。
使用fopen
函数,由于在用户态下就有了缓冲,因此进行文件读写操作的时候就减少了用户态和内核态的切换(切换到内核态调用还是需要调用系统调用API:read
,write
);而使用open
函数,在文件读写时则每次都需要进行内核态和用户态的切换;表现为,如果顺序访问文件,fopen
系列的函数要比直接调用open
系列的函数快;如果随机访问文件则相反。
这样一总结梳理,相信大家对于两个函数及系列函数有了一个更全面清晰的认识,也应该知道在什么场合下使用什么样的函数更合适,效率更高。
标准c库打开创建文件读写文件光标移动
FILE *fopen(char *filename, char *mode)
filename 文件名称
mode 打开模式:
r 只读方式打开一个文本文件
rb 只读方式打开一个二进制文件
w 只写方式打开一个文本文件
wb 只写方式打开一个二进制文件
a 追加方式打开一个文本文件
ab 追加方式打开一个二进制文件
r+ 可读可写方式打开一个文本文件
rb+ 可读可写方式打开一个二进制文件
w+ 可读可写方式创建一个文本文件
wb+ 可读可写方式生成一个二进制文件
a+ 可读可写追加方式打开一个文本文件
ab+ 可读可写方式追加一个二进制文件
size_t fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream);
size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
int fseek(FILE *stream, long offset, int whence);
ptr:缓冲区 size:一个字符的大小 nmemb:个数 stream:哪个文件
fwrite:在字符串 ptr ,写 size 字符大小,写 nuemb 次,写道 stream 文件里面。
fread:在stream文件里面,读 size 字符大小,读 nuemb 次,读到 ptr 缓冲区
#include <stdio.h>
#include <string.h>
int main(){
FILE *fd;
char *str = "Hello world !";
char readbuf[128] = {0};
fd = fopen("./file4","w+");
// size_t fwrite(const void *ptr, size_t size, size_t nmemb,FILE *stream);
fwrite(str,sizeof(char),strlen(str),fd);
// int fseek(FILE *stream, long offset, int whence);
fseek(fd,0,SEEK_SET);
// size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
fread(readbuf,sizeof(char),strlen(str),fd);
printf("read %s\n",readbuf);
fclose(fd);
return 0;
}
fwrite的返回值取决于第三个参数nmemb
fread的返回值与fwrite区别于fread受所读内容大小的限制。
#include <stdio.h>
#include <string.h>
int main(){
FILE *fd;
char *str = "Hello world !";
char readbuf[128] = {0};
fd = fopen("./file4","w+");
// size_t fwrite(const void *ptr, size_t size, size_t nmemb,FILE *stream);
int nwrite = fwrite(str,sizeof(char),100,fd);
// int fseek(FILE *stream, long offset, int whence);
fseek(fd,0,SEEK_SET);
// size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
int nread = fread(readbuf,sizeof(char),strlen(str),fd);
printf("read %s\n",readbuf);
printf("read:%d,write:%d\n",nread,nwrite);
fclose(fd);
return 0;
}
fputc的用法
一个个字符地写入文件。
int fputc(int c, FILE *stream);
c:要写入的内容 stream:文件描述符
#include <stdio.h>
#include <string.h>
int main(){
FILE *fp;
char *str = "Hello world !";
fp = fopen("./text","w+");
int len = strlen(str);
int i;
for(i = 0;i < len;i++){
fputc(*str,fp);
str++;
}
fclose(fp);
return 0;
}
fgetc的用法
一个个字符从文件里读出来。
int fgetc(FILE *stream);
feof的用法
int feof(FILE *stream);
未到达文件尾巴时返回值为0,到达文件尾巴时返回整数。
#include <stdio.h>
#include <string.h>
int main(){
FILE *fp;
fp = fopen("./text","r");
char c;
while(!feof(fp)){
c = fgetc(fp);
printf("%c",c);
}
printf("\n");
fclose(fp);
return 0;
}