底层文件访问:
进程:运行中的程序,它有一些与值关联的文件描述符,有多少个文件描述符取决于系统配置情况。
当一个程序开始运行时,一般会打开三个文件描述符:
0:标准输入
1:标准输出
2:标准错误
可以通过系统调用(open)把其他文件描述符与文件和设备相关联。
write系统调用:
作用:把缓冲区(buf)的前nbytes个字节写入与文件描述符(fildes)关联的文件。
write返回实际写入的字节数,如果文件描述符有错误或者底层设备的驱动程序对数据长度比较敏感,表示在write调用中出现了
错误,返回值可能会小于nbytes。如果函数返回0,表示未写入任何数据;返回-1表示write调用中出现了错误,错误代码保存在
全局变量errno中。
#include <stdio.h> #include <unistd.h> // write调用原型所在的头文件 int main(void) { if ( (write(1, "Here is some data ", 18)) != 18 ) write(2, "A write error has occurred on file descriptor 1 ", 46); return 0; }
root@ubuntu:/home/linlin/linlin/c_code# ./simple_write.out
Here is some data
write可能会报告写入的字节比要求的少,这不一定时错误,在程序中,需要检查errno以发现错误,然后再次调用write写入剩
余数据
read系统调用:
作用:从文件描述符(fildes)相关联的文件里读入nbytes个字节的数据,并把它们放到数据区buf中。
read返回实际读入的字节数,这可能会小于请求的字节数,如果read调用返回0,表示未读入任何数据,已到达了文件尾;如果返
回-1,表示read调用出现了错误。
#include <stdio.h> #include <unistd.h> int main(void) { char buffer[128]; int nread; nread = read(0, buffer, 128); if (-1 == nread) write(2, "A read error has occurred ", 26); if ( (write(1, buffer, nread)) != nread ) write(2, "A write error has occurred ", 27); return 0; }
root@ubuntu:/home/linlin/linlin/c_code# echo hello there | ./simple_read.out
hello there
open调用:
作用:创建一个新的文件描述符。
#include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> int open(const char *pathname, int flags); int open(const char *pathname, int flags, mode_t mode);
open建立了一条到文件或设备的访问路径,如果调用成功,它可以返回一个可以被read,write和其它系统调用使用的文件描述符
这个文件描述符时唯一的,它不会与任何其他运行中的进程共享。
准备打开的文件或设备的名字作为参数path传递给函数,oflags参数用于指定打开文件所采取的动作。
oflags参数是通过必须文件访问模式与其它可选模式相结合的方式指定的:
模式 说明
O_RDONLY 以只读方式打开
O_WRONLY 以只写方式打开
O_RDWR 以读写方式打开
open调用还可以在oflags参数中包含可选模式的组合(用按位或操作):
O_APPEND: 把写入数据追加在文件末尾
O_TRUNC:把文件长度设置为0,丢弃已有的内容
O_CREAT: 如果需要,就按照参数mode中给出的访问模式创建文件
O_EXCL:与O_CREAT一起使用,确保调用者创建出文件。open调用是一个原子操作,它只执行一个函数调用。使用这个可选模
式可以防止两个程序同时创建同一个文件。如果文件已存在,open调用将失败。
访问权限的初始值:
当使用代用O_CREAT标志的open调用来创建文件时,必须使用带有三个参数格式的open调用,第三个参数mode是几个标志按位或得
到的,这些标志在sys/stat.h中定义:
S_IRUSR: 读权限,文件属主
S_IWUSR:写权限,文件属主
S_IXUSR:执行权限,文件属主
S_IRGRP: 读权限,文件属组
S_IWGRP:写权限,文件属组
S_IXGRP:执行权限,文件属组
S_IROTH: 读权限,其他用户
S_IWOTH:写权限,其他用户
S_IXOTH:执行权限,其他用户
#include <stdio.h> #include <fcntl.h> #include <sys/types.h> #include <sys/stat.h> int main(void) { open("test", O_CREAT, S_IRUSR|S_IXOTH); return 0; }
文件访问权限的影响因素:
指定的文件权限只有在创建文件时才会使用
umask(用户掩码)会影响到被创建文件的访问权限,open调用中给出的mode值将与当时的用户掩码的反值做AND操作
例如umask为001,open调用指定了S_IXOTH模式标志,那么创建的文件对其他用户不会拥有写执行权限。
open和creat调用中的标志实际上是发出设置文件访问权限的请求,所请求的权限是否会被设置取决于当时的umask值。
复制文件的两个不同实现:
#include <unistd.h> #include <sys/stat.h> #include <fcntl.h> int main(void) { char block[1024]; int in, out; int nread; in = open("file.in", O_RDONLY); out = open("file.m.out", O_WRONLY|O_CREAT, S_IRUSR|S_IWUSR); while ((nread = read(in, block, sizeof(block))) > 0) write(out, block, nread); return 0; }
#include <unistd.h> #include <sys/stat.h> #include <fcntl.h> int main(void) { char c; int in, out; in = open("file.in", O_RDONLY); out = open("file.out", O_WRONLY|O_CREAT, S_IRUSR|S_IWUSR); while (read(in, &c, 1) == 1) write(out, &c, 1); return 0; }
close系统调用:
使用close调用终止文件描述符fildes与其对应文件之间的关联。文件描述符能被释放并能够重新使用。
close调用成功时返回0,出错时返回-1
#include <unistd.h> int close(int fildes);
检查close调用的返回结果非常重要,有的文件系统,特别是网络文件系统,可能不会再关闭文件之前报告文件写操作中
出现的错误,这是因为在执行操作时数据可能未被正确写入。
ioctl系统调用:
作用;提供一个控制设备及其描述符行为和配置底层服务的接口。
终端,文件描述符,套接字设置磁带机都可以有为它们定义的ioctl。
#include <unistd.h> int ioctl(int fildes, int cmd, ...);
ioctl对描述符fildes引用的对象执行cmd参数中给出的操作。根据特定设备支持操作的不同,它可能会有一个可选的第三参数。