1、fopen和fclose
fopen是C的标准库函数,用来打开文件
open和fopen的区别
1、open是Unix系统调用函数(包括Linux),返回的是文件描述符(File Descriptor),它是文件在文件描述符表里的索引
fopen是ANSIC标准中的C语言库函数,在不同的系统中应该调用不同的内核API,返回的是一个指向文件结构的指针
2、“fopen”是C标准函数,因此具有很好的移植性,而“open”是Unix,Linux系统调用,移植性有限,
3、open返回的是文件描述符,而文件描述符是Unix和Linux系统下一个非常重要的东西,Unix和Linux系统一切设备都是以文件的形式操作,如网络套接字,硬件设备等,当然也包括操作普通正规的文件
fopen是用来操作普通文件的
4、如果从文件IO的角度来看,open属于低级IO函数,fopen属于高级IO函数,低级和高级的简单区分标准是谁离系统内核更近,低级文件IO运行在内核态,高级文件IO运行在用户态。
缓冲文件系统和非缓冲文件系统
a:缓冲文件系统
缓冲文件系统的特点是:在内存开辟一个“缓冲区”,读文件时,将磁盘文件的数据先读入内存“缓冲区”,装满后再从内存“缓冲区”读出数据。写文件时,先将数据写入内存“缓冲区”,待内存“缓冲区”装满后再写入文件。可以看出,内存“缓冲区”的大小,影响着实际操作外存的次数,内存“缓冲区”越大,则操作外存的次数就少,执行速度就快、fopen, fclose, fread, fwrite, fgetc, fgets, fputc, fputs, freopen, fseek, ftell, rewind等。
b:非缓冲文件系统
非缓冲文件系统是借助文件结构体指针来对文件进行管理,通过文件指针来对文件进行访问,既可以读写字符、字符串、格式化数据,也可以读写二进制数据。非缓冲文件系统依赖于操作系统,通过操作系统的功能对文件进行读写,是系统级的输入输出,它不设文件结构体指针,只能读写二进制文件,但效率高。open, close, read, write, getc, getchar, putc, putchar等。
简单来说就是open无缓冲,fopen有缓冲。open与read, write等配合使用, fopen与fread,fwrite等配使用
头文件
#include <stdio.h>
函数原型
FILE *fopen(const char *path, const char *mode);
FILE:文件类型指针(文件指针),每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的有关信息(文件的名字,状态,文件的当前位置等),这些信息是保存在一个结构体里面,该结构体的类型是由系统声明的,取名为FILE, FILE *fp:定义一个FILE类型的指针。
path:要打开文件的文件名或者文件的路径
mode:表示打开文件的权限,也就是文件的使用方式
下面是常用的权限
fclose
头文件和fopen相同
函数原型
int fclose(FILE *fp);
fclose函数用来将fp指针指向的文件关闭
2、fread和fwrite
头文件
#include <stdio.h>
函数原型
size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
size_t fwrite(const void *ptr, size_t size, size_t nmemb,FILE *stream);
ptr:一个地址,对fread来说,就是用来存放从文件读取的数据地址,对fwrite来说就是将地址中的数据写入到文件
size:要读写的字节数
nmemb:要读取多少个数据帧(一个数据帧长度为size)
stream:FILE类型的指针,指向要读写的文件
3、fseek函数
头文件和fopen一样
函数原型
int fseek(FILE *stream, long offset, int whence);
stream:文件类型的指针,表示要操作的文件
offset:偏移量
whence:起始点
对于fopen、fwrite、fread、lseek、fclose的使用
#include <string.h>
#include <stdio.h>
int main()
{
FILE *fp;
char* str = "hello world!";
char readBuf[128] = {0};
fp = fopen("./file1.txt","w+");
fwrite(str,sizeof(char),strlen(str),fp); //表示读取strlen(str)次,一次为一个char的大小
// fwrite(str,sizeof(char)*strlen(str),1,fp); 表示读取一次,文件大小为sizeof(char)*strlen(str)
fseek(fp,0,SEEK_SET);
fread(readBuf,sizeof(char),strlen(str),fp);
// fread(readBuf,sizeof(char)*sizeof(str),1,fp);
printf("read data: %s\n",readBuf);
close(fp);
return 0;
}
可以看到程序运行后创建了一个file1.txt文件,并且将字符串写入到文件,再将字符串读出
当然也可以将结构体写入到文件
#include <string.h>
#include <stdio.h>
struct Test{
int a;
char c;
};
int main()
{
FILE *fp;
struct Test data1 = {100,'w'};
struct Test data2;
fp = fopen("./file1","w+");
fwrite(&data1,sizeof(struct Test),1,fp); //读取一次,大小为一个结构体的大小
fseek(fp,0,SEEK_SET);
fread(&data2,sizeof(struct Test),1,fp);
printf("read2 data:%d %c\n",data2.a,data2.c);
close(fp);
return 0;
}
4、fputc和fgetc
头文件和fopen函数一样
函数原型
int fputc(int c, FILE *stream);
int fputs(const char *s, FILE *stream);
int fgetc(FILE *stream);
例
fputc:
#include <string.h>
#include <stdio.h>
int main()
{
FILE *fp;
int i;
char *str = "hello world";
int len = strlen(str);
fp = fopen("./file1","w+");
for(i = 0;i < len;i++){
fputc(*str,fp);
str++;
}
close(fp);
return 0;
}
编译运行后file1中
fgetc和feof:
int main()
{
FILE *fp;
int i;
char c;
fp = fopen("./file1.txt","r");
while(!feof(fp)){
c = fgetc(fp);
printf("%c",c);
}
fclose(fp);
return 0;
}
feof是C语言标准库函数,其头文件为stdio.h,其功能是检测流上的文件结束符,如果文件结束,则返回非0值,否则返回0(即,文件结束:返回非0值;文件未结束:返回0值)。
5、fgets和fputs
头文件和fopen相同
函数原型
char *fgets(char *s, int size, FILE *stream);
int fputs(const char *s, FILE *stream);
n-1:先从文件里面读入n-1个字符,然后在最后加上“\0”这样就得到了n个字符,再放入到str中
对于fputs,字符串末尾的“\0”不输出
fputs和fgets函数的功能相当于gets和puts函数,只是gets和puts是以终端为读写对象,而fgets和fputs函数是以指定文件作为读写对象