C标准库函数是工作在系统库函数之上的。C标准库函数在读写文件时候都有一个文件流指针。
当我们调用C标准库的printf函数时,printf函数实际上不知道如何显示字符到显示屏上,而是去调用应用层的writeAPI函数接口,然后应用层的API函数会自动去调用内核sys_write函数,最后送到显示屏上显示。
当我们打开一个文件时,实际上会返回FILE*结构指针,FILE* fp=fopen(F_PATH,"r");
FILE文件流指针的结构体如图所示。
FILE结构指针包含三个信息,文件描述符,文件读写的位置f_pos,写多大的buffer。
文件描述符指向真正 的磁盘文件,在进行文件读写操作时候是先读写到缓冲区,然后再调用系统应用层API write函数进行写操作,write将文件内容写到内核缓冲区,然后再调用内核层API sys_write进行写操作。减少I/O,提高效率。
当我们使用C语言标准库函数fopen()每打开一个文件时候,其都会对应一个单独一个缓冲区。内核缓冲区是公用的。
C标准库函数是有缓冲区,当我们往磁盘上写入数据时,会先写入buffter缓冲区,然后buffer缓冲区会根据自己的刷新方式,一次性把数据送出去
刷新缓冲区:一般使用fflush()函数去刷新,当缓冲区满、程序的正常结束、以及fclose操作等都会刷新缓冲区。换行符\n 只能刷新终端文件的缓冲区
下面以一个读写文件为例:
1. 将hello world字符写入磁盘hello.txt 文件中
2. 使用fopen打开hello.txt 文件,然后进行写操作
3. 写到C标准缓冲区
4. 满足刷新缓冲区条件时,会调用系统应用层API write函数进行写操作
5. write将文件内容写到内核缓冲区
6. 如果内核缓冲区没有满,系统不会立即调用内核层API sys_write将缓冲区内容写入到磁盘,有一个守护进程会定时刷新内核缓冲区。
7. 此时有一个进程B读hello.txt这个文件,发现内核缓冲区就有这个文件内容,其就不需要访问hello.txt 磁盘文件了。
具体流程如下: