文件读写基本(open详解,及文件描述符)
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>
int main(void){
int fd=0;
fd=open("./file.txt",O_RDWR);
if(-1 == fd){
fprintf(stderr, "%s\n",strerror(errno));
exit(1);
}
printf("open ok\n");
char buf1[]="hello word";
write(fd,buf1,strlen(buf1));
lseek(fd,SEEK_SET,0);
char buf2[30]={0};
read(fd,buf2,sizeof(buf2));
close(fd);
printf("buf2=%s\n",buf2 );
}
open函数※
a.利用文件名,通过文件系统找到块设备(磁盘,U盘等)上的文件
文件系统就是程序代码,组织管理块设备上的所有文件
文件系统属于OS的一部分
b.找到文件后,调用块设备驱动程序,打开文件
驱动程序也是os的一部分
open函数使用※
man 2 open
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
int open(const char *pathname, int flags);
功能:只能打开已经存在的文件,如果不存在就返回-1 报错
int open(const char *pathname, int flags, mode_t mode);
功能:如果文件存在就直接打开,不存在就按照指定的文件权限,创建指定名字的新文件
三个参数时,可以打开已存在的文件和文件不存在时创建文件
open返回值:
成功返回一个非负整数的文件描述符
失败返回-1,并且设置错误号给系统定义的全局变量errno,用于标记函数到底出了什么错
flags参数:model,创建文件时,用于指定文件的原始权限,就是rwxrwxr--
flags:O_RDONLY,O_WRONLY,O_RDWR,O_TRUNC,O_APPEND
指定打开文件的方式,这些宏可以使用|组合,可同时指定多个
宏的含义
a. O_RDONLY 只读方式打开,只能对文件进行读
b. O_WRONLY 只写方式打开,只能对文件进行写
c. O_RDWR 可读可写方式打开文件
以上三个在指定时,只能唯一指定,不可以组合,比如O_RDONLY|O_WRONLY
d. O_TRUNC:打开时将文件内容清0
e. O_APPEND: 打开文件后,追加新数据到文件末尾
不指定这个选项的话,新写入的数据会从头开始写,覆盖原有数据
f. O_NONBLOCK,O_ASYNC
a,b,c不可以组合,但可以和def组合
g. O_CREAT
指定时,文件存在,直接打开
文件不存在,创建该名字的文件
指定此参数时,需要设置第三个参数,用于指定新打开文件的原始权限
h. O_EXCL
当O_EXCL与O_CREAT同时被指定,打开文件时,如果文件存在,就报错
单独指定没有意义,需和O_CREAT配合使用
意义:保证每次open的是一个新文件,如果文件之前就存在,提醒打开的不是一个新文件
保证检查文件的存在和创建文件是原子操作。
对于打开软链接,不加O_EXCL会在指向的位置创建文件,加上O_EXCL可禁止在指向位置创建文件,调用失败,避免特权应用在系统目录下创建文件,消除安全隐患
open打开成功后,open做了哪些事情※
1.记录打开文件的信息
a.程序运行起来后就是一个进程,OS会创建一个task_struct结构体,记录程序运行时的各种信息,
比如所打开文件的相关信息
b.open将文件成功打开后,在task_struct中又会创建一些结构体(数据结构),
用于记录当前进程目前所打开的文件信息,后续所有的文件操作,都需要依赖于这些信息,
其中包括指向打开文件的文件描述符
2.open函数会申请一段内存空间(内核缓存),后续读写文件时,用于临时缓存读写文件时的数据
a.缓存:
即开辟的一段内存空间,比如char buf[100]就是一段100字节的缓存空间,用于临时存放中转的数据
b.内核缓存与应用缓存:
open是OS所提供的新系统函数,属于OS内核的一部分,所以open函数开辟的缓存空间就是内核缓存
比如代码里buf1和buf2是应用程序定义的,因此叫应用缓存
其实定义的一切变量空间,比如 Int a,int b[10] 都是缓存,用于临时缓存数据
c.open为什么要开辟内核缓存空间
内核读写速度>磁盘读写速度,有了在内存中开辟的内核缓存后,上层读写数据时,会直接读写缓存,速度会很快,
至于缓存与磁盘文件数据的交换,就留给下层去做,这样可以节省上层操作的时间
d.注意:open时只是开辟了内核缓存空间,里面并没有数据,只有当进行读写时,才会缓存读写数据
e.缓存数据的流动
写:应用缓存->open函数->写到内核缓存--->驱动程序(驱动缓存)->磁盘上
读:磁盘->驱动缓存->内核缓存->应用缓存->打印输出
文件描述符※
open成功就会返回一个非负整数(0,1,2,3 ....)的文件描述符,
文件描述符指向打开的文件,后续的read/write/close等函数的文件操作,都是通过文件描述符来实现
文件描述符池
每个程序运行起来后,就是一个进程,系统会给每个进程分配0~1023的文件描述符范围,
每个进程打开文件时,open所返回的文件描述符是0~1023范围中的某个数字
0~1023 这个范围,就是文件描述符池
1023这个上限可以修改,但已经够用,一个进程同时打开1023个文件的情况很少见,所以一般很少改
open返回文件描述符的规则:返回当前最小没用的那个
进程一运行起来,0,1,2已经被使用了,最小没被用的是3,所以返回3
如果又打开了一个文件,最小没被用的应该是4
文件关闭后,被文件用的描述符会被释放,等着下一次open时被重复利用
下面的2次均为3
#include <fcntl.h>
#include <stdio.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
int main(int argc,char *argv[]){
int fd=open("./1.txt",O_RDWR|O_CREAT,0664);
if(fd<0){
fprintf(stderr, "err:%s\n",strerror(errno));
}else{
printf("fd=%d\n",fd );
}
close(fd);
fd=open("./11.txt",O_RDWR|O_CREAT,0664);
if(fd<0){
fprintf(stderr, "err:%s\n",strerror(errno));
}else{
printf("--fd=%d\n",fd );
}
close(fd);
return 0;
}
open文件描述符与fopen文件指针关系※
1.open:linux的系统函数(文件Io函数)
man 2 open
open成功后,返回的文件描述符,指向打开的文件
2.fopen: c库的标准io函数
man 3 fopen
#include <stdio.h>
FILE *fopen(const char *path, const char *mode);
fopen成功后,返回的是FILE *的文件指针,指向了打开的文件
3.对于linux的c库来说,fopen这个c库函数,最终其实还是open函数来打开文件的
fopen只是对open这个函数的二次封装
应用程序->FILE * fopen标准io函数(c库函数)->int open文件io函数(linux系统函数)
fopen的文件指针,最终还是会被替换成open的文件描述符,然后用于操作打开的文件