3- 文件IO
本章说明了可用的文件IO函数 :打开文件,读文件,写文件等。
UNIX系统中的大多数文件I/O只需5个函数 :open,read,write,lseek,以及close.
然后说明不同缓冲长度对read,write 的影响。
本章描述的函数常被称为不带缓冲的I/O (指read和write都调用内核中的一个
系统调用(内核提供的、功能十分强大的一系列的函数。这些系统调用是在内核中实现的,
再通过一定的方式把系统调用给用户,一般都通过门(gate)陷入(trap)实现。系统调用是用户程序和内核交互的接口。))
只要涉及多个进程共享资源,原子操作的概念就非常重要,通过文件I/O和open函数的参数讨论了此概念。
然后进一步说明讨论了多个进程如何共享文件。以及所涉及的内核有关数据结构,最后将说明 dup,fcntl,sync,fsync,ioctl函数。
文件描述符:对内核而言所有打开的文件都通过问卷描述符引用,文件描述符是一个非负整数。
当打开一个现有文件或创建一个新文件时,内核向进程返回一个文件描述符。
按照惯例UNIX系统shell把文件描述符0,1,2分别与标准输入,输出,错误关联。
3.3open和openat
3.4creat
3.5close
3.6lseek
3.7read
3.8write
3.10文件共享:
内核用三种数据结构打开文件:
1.每个进程在进程表中有一个记录项 ,记录项中包含一张打开de 文件描述符表 ,可将其视为一个矢量 ,每个描述符占用一项 :
a. 文件描述符标志
b. 指向一个文件表项的指针
2. 内核为所有打开de 文件 维持一张文件表,每个文件表项包含:
a. 文件状态标志(读,写,追写,同步,非组赛等).
b. 当前文件偏移量.
c. 指向该文件v节点的表项指针
3.每个打开文件都有一个v节点结构。V节点包含了文件类型和对此文件进行各种操作的函数指针 ;
对于大多数文件v节点还包含了该文件的i节点(i-node索引节点)
3.11原子操作:由多步组成一个操纵,如果该操作原子的执行 要么执行完所有步骤,要么一步也不执行。
3.12dup和dup2
3.13sync,fsync,fdatasync
3.14fcntl
例程 :cp :
/*
* 将标准输入 复制到标准输出 。
*/
#include "apue.h"
#define BUFFSIZE 4096
/*
*书P59 显示了用 20种不同的缓冲区长度 读516581760字节所占用cpu的时耗。
*此测试所用文件系统是linux ext4系统 ,其磁盘块长度为4096字节 ,
*图中系统cpu时间的最小值差不多出现在bufsize为4096及以后位置.继续增加缓冲区长度对此时间几乎没影响 .
*/
int
main(void)
{
int n;
char buf[BUFFSIZE];
while((n = read(STDIN_FILENO,buf,BUFFSIZE))>0)
{
if(write(STDOUT_FILENO,buf,n)!=n)
printf("write error\n");
}
if(n==-1) printf("read error\n");
return 0;
}