Unix环境高级编程笔记:3、文件I/O
程序员文章站
2022-05-07 18:18:14
...
1、文件io需求5个函数
open read write lseek close
这些函数被称为不带缓冲的IO,不带缓冲IO指的是每个read和write都调用内核中的系统调用
只要涉及多个进程间共享资源,原子操作的概念就变的非常重要。
dup、fcntl、sync、fsync、ioctl
2、文件描述符
文件描述符是一个非负整数,当打开一个现有的文件或创建一个新文件时,内核向进程返回一个文件描述符。
<unistd.h>
0 标准输入 STDIN_FILENO
1 标准输出 STDOUT_FILENO
2 标准出错输出 STDERR_FILENO
3、open
open函数可以打开或创建一个文件
#include <fcntl.h>
int open(const char * pathname,int oflag,.../*mode_t mode*/);
oflag参数说明此函数多个选项,用下面一个或多个常量进行“或”运算构成oflag参数 <fcntl.h>
O_RDONLY 只读打开
O_WRONLY 只写打开
O_RDWR 读写打开
O_APPEND 写时追加到文件尾端
O_CREAT 或此文件不存在则创建。使用此选项,需要第三个参数mode,用其指定新文件的访问权限位
O_EXCL
O_TRUNC 如果此文件存在,而且为只写或读写成功打开,则将其长度截短为0
O_NOCTTY
O_NONBLOCK 设置非阻塞IO
4、 creat
#include <fcntl.h>
int creat(const char * pathname,mode_t mode);
5、close
#include <unistd.h>
int close(int fildes); 成功0,出错-1
6、lseek
按系统默认情况,当打开一个文件时,除非指定O_APPEND选项,否则该偏移量被设置为0
lseek显式的为一个打开的文件设置其偏移量
off_t lseek(int fildes,off_t offset,int whence);
whence是SEEK_SET,则将该文件的偏移量设置为距文件开始处offset个字节
whence是SEEK_CUR,则将该文件的偏移量设置为其当前值加offset,offset可为正或负
whence是SEEK_END,则将该文件的偏移量设置为文件长度加offset,offset可为正或负
7、read
#include<unistd.h>
ssize_t read(int filedes,void *buf,size_t nbytes) 成功返回读到的字节数,已到文件结尾返回0,出错返回-1
8、write
#include<unistd.h>
ssize_t write(int filedes,const void *buf,size_t nbyes); 成功则返回已写的字节数,若出错则返回-1
stdin.c 标准输入写到标准输出
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <stdlib.h>
#define BUFFER_SIZE 4096
int main(int argc, char **argv) {
char buffer[BUFFER_SIZE];
int n;
while ((n = read(STDIN_FILENO, buffer, BUFFER_SIZE)) > 0) {
if (write(STDOUT_FILENO, buffer, n) != n) {
printf("write error");
exit(1);
}
}
if (n < 0) {
printf("read error");
exit(1);
}
exit(0);
}
|
9、dup dup2
#include <unistd.h>
int dup(int fileds);
int dup2(int files,int files2);
dup返回的新文件描述符一定是当前可用的文件描述符中最小的值。dup2可以用fileds2参数指定新描述符的数值
当此函数执行时,二个描述符指向同一个文件表项,所以他们共享同一文件状态标志(读 写 添加)以及同一个文件偏移量。
10、fsync fdatasync sync
为保证磁盘上的实际文件与缓冲区内容的一致性
11、fcntl
#include<fcntl.h>
int fcntl(int filedes,int cmd,.../*int arg*/);
fcntl函数有5种功能
a)复制一个现有的描述符(cmd=F_DUPFD)
b)获得/设置文件描述符标记(cmd=F_GETFD或F_SETFD)
c)获得/设置文件状态标志(cmd=F_GETFL或F_SETFL)
d)获得/设置异步IO所有权(cmd=F_GETOWN或F_SETOWN)
e)获得/设置记录锁(cmd=F_GETTLK、F_SETLK、F_SETLKW)
fcntl文件状态标志
O_RDONLY 只读打开
O_WRONLY 只写打开
O_RDWR 读写打开
O_APPEND 写时追加到文件尾端
O_CREAT 或此文件不存在则创建。使用此选项,需要第三个参数mode,用其指定新文件的访问权限位
O_EXCL
O_TRUNC 如果此文件存在,而且为只写或读写成功打开,则将其长度截短为0
O_NOCTTY
O_NONBLOCK 设置非阻塞IO
fcntl.c
#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
int main(int argc, char **argv) {
int val;
if(argc !=2) {
printf("argc !=2\n");
exit(1);
}
if((val = fcntl(atoi(argv[1]),F_GETFL,0)) < 0) {
printf("fcntl error");
exit(1);
}
switch (val & O_ACCMODE) {
case O_RDONLY:
printf("read only");
break;
case O_WRONLY:
printf("write only");
break;
case O_RDWR:
printf("read write");
break;
default:
printf("unknow acces mode");
}
if(val & O_APPEND)
printf("append");
if(val & O_NONBLOCK)
printf("nonblocking");
#if defined (O_SYNC)
if(val & O_SYNC)
printf("synchronous");
#endif
putchar('\n');
exit(0);
}
|
12、ioctl
ioct函数是io操作的杂物箱,不能使用本中其他函数表示的io操作通用都能用ioctl表示。终端io是ioctl的最大使用方面。
13、/dev/fd
/dev/fd目录,其目录项是名为0、1、2等文件,打开文件/dev/fd/n 等效于复制描述符n
fd = open("/dev/fd/0",mode);
等效于fd = dup(0)
所以描述符0和fd共享同一文件表项,例如,描述符0先前被打开为只读,那么我们也只能对fd进行只读操作