基础IO:文件的输入输出操作–回顾标准库的IO接口/学习linux下系统调用IO接口/文件描述符的理解/重定向的理解/文件系统/库文件
fopen/fwrite/fread/fseek/fclose stdin/stdout/stderr
FILE fopen(charfilename,char mode);(文件名称,打开方式)
打开方式:“r-只读 r±读写 w-只写 w±读写 a-追加写 a±追加读写 b-二进制操作"
追加写:每次写入数据总是写入到文件末尾
r+的读写和w+的读写有什么区别??:r+读写打开文件;若文件不存在则报错/w+读写打开文件,若不存在则创建,若存在清空原有内容
a-不仅仅是追加写,并且文件不存在还会创建新文件
b-默认清空如果不指定b,则认为文件是文本操作,加上b则认为是二进制操作;区别在于-有时候一个特殊字符,只是一个字符但是占据两个字节的内存(读取一个100字节大小的文件,文本操作最终读取出来的数据,不一定是100字节)
返回值:返回一个FILE的文件流指针作为文件的操作句柄;失败返回NULL
size_t fread(char buf,size_t block_size,size_t block_count,FILEfp);(缓冲区,块大小,块个数,文件流指针)
size_t fwrite(char* data,size_t block_size,size_t block_count,FILE*fp):(数据首地址,块大小,块个数,文件流指针);
注意:fread/fwrite 操作的数据实际大小是块大小*块个数; 块大小=10.块个数=2; 意味着要写入/读取20个字节的数据;
返回值:返回实际操作的块个数
例如:读取一个文件size=10;count=2;
如果文件大小足够则返回2
但是若文件大小只有16字节,则会返回1,因为第二块没有读满
fread如果读到了文件末尾会返回0;若读取1000个字节块个数为1,文件大小只有512字节,虽然读取了512数据,但是依然会返回0;
int fseek(FILE*fp,long offset,int whence);(将文件的读写指针从whence位置偏移offset个字节)–跳转文件读写位置
SEEK_SET-从文件起始偏移
SEEK_CUR-从当前读写位置开始偏移
SEEK_END-从文件末尾偏移
int fclose(FILE *fp):关闭文件流指针,释放资源
fread/fwrite比较推荐块大小为1,块个数是想要操作的数据长度
fseek:文件没有数据也可以跳转读写位置;可以向后任意跳转文件的当前读写位置
对文件数据进行字符串操作的时候要注意文件数据中的\0这种数据
打开文件一定不要忘了关闭文件
标准库中的IO接口–都是库函数–而库函数就是对系统调用接口的一层封装
系统调用IO接口的学习:open/read/write/lseek/close
#include
filename:要打开的文件名称
flag:选项参数–文件的打开方式 必选项/可选项
必选项:O_RDONLY-只读 /O_WRONLY-只写 / O_RDWR-读写 不选的话默认只读
可选项:O_CREAT-文件存在则打开,不存在则创建;O_EXCL与O_CREAT同时使用,文件存在则报错;O_TRUNC-打开文件的同时清空原有内容;O-APPEND-追加写,总是将数据写入到文件末尾
mode:权限—如果使用了O_CREAT有可能创建新文件,就一定要指定文件权限(一定要通过第三个参数指定文件的权限(mode&~ umask)-----umask(0),八进制数字形式
返回值:一个非负整数—文件描述符—文件的操作句柄;失败返回-1;
fd:open返回的文件描述--文件操作句柄
buf:要写入文件的数据的空间首地址
count:要写入的数据大小
返回值:返回实际写入文件的数据字节长度;失败返回-1;
ssize_t write(int fd,char* data,size_t len);
fd:open返回的文件描述—文件的操作句柄–通过这个fd指定要往哪个文件写入数据 data:要写入文件的数据的首地址
len:要写入文件的数据长度
返回值:返回实际写入文件的数据字节长度,错误返回-1;
fd:open返回的文件描述符
buf:从文件中读取数据放到哪块缓冲区中的首地址
len:想要读取的数据长度,注意这个len不能大于缓冲区的大小
返回值:返回的是实际读取到的数据字节长度,错误返回-1;
fd:open返回的文件描述符
offset:偏移量
whence:从哪里开始偏移 SEEK_SET-文件起始位置 SEEK_CUR-文件当前读写位置 SEEK_END-文件末尾 返回值:成功返回当前位置相当于起始位置的偏移量;失败返回-1;
其实就是内核中一个进程打开的文件描述信息表的**下标**--通过这个下标可以在内核中找到相应的文件描述信息,通过这个描述信息可以实现文件的操作
**为什么打开一个文件,如果不操作了一定要关闭,释放资源?
文件描述符实际是有限的;若不关闭文件,文件描述符用光,则在进程中就打不开新文件了**
一个程序运行起来,进程中会默认打开三个文件:标准输入-0-stdin/标准输出-1-stdout/标准错误-2-stderr
文件描述符有个分配规则:最小未使用;
printf打印数据到标准输出,close(1),就是把标准输出关闭了;打开新文件后,printf并没有把数据打印出来,而是在刷新缓冲区fflush(stdout)后,将数据写入到了文件中。
printf并非真的一定要把数据写入标准输出文件,而是因为printf函数中操作文件的时候操作的描述符是1,原本向1中写入数据,就是向标准输出写入,然后当1指向了新的文件后,这时候printf就会将数据写入到制定新的文件中。(重定向原理)
重定向:将数据不在写入原本的文件,而是写入新的指定的文件中–实现方式就是替换这个描述符对应的文件描述信息
int dup2(int oldfd,int newfd)—描述符重定向
例:dup2(fd,1);//将1重定向到test.txt这个文件
ls -l >a.txt
ls -l --依然使用之前的解析方案
清空重定向 open(O_CREAT|O_TRUNC)
追加重定向 open(O_CREAT|O_APPEND)
a.txt fd=open(a.txt); dup2(fd,1); 子进程在运行指令的时候ls本身要将数据写入标准输出
文件描述符与文件流指针的关系:
文件描述符:是一个非负整数----系统调用的IO接口
文件流指针:FILE结构体–typedef struct _IO_FILE FILE --库函数IO接口的操作句柄
通过文件流指针进行最终文件操作的时候,依然还要能够找到文件对应的文件描述符才可以,文件流指针是一个结构体,结构体中有很多成员变量,其中就有一个叫_fileno–这就是文件描述符
代码链接: 添加重定向的minishell.
向文件写入数据,并不会直接写入文件,而是先写入缓冲区中,刷新缓冲区的时候才会写入文件;
系统调用接口是直接将数据写入文件的,系统调用接口是没有这个缓冲区的,只有库函数才存在这个缓冲区
举例:exit退出会刷新缓冲区/_exit退出时不会刷新缓冲区