fcntl函数可以改变或者查看已打开文件的性质。该函数的定义如下:
- #include <fcntl.h>
- int fcntl(int fd, int cmd);
- int fcntl(int fd, int cmd, long arg);
- int fcntl(int fd, int cmd, struct flock *lock);
-
本文只介绍函数的第1,2种形式,第3中形式会在以后介绍记录锁时说明。
fcntl有5种功能:
1.复制一个现有的描述符。(cmd = F_DUPFD)
2.获取/设置文件描述符标记。(cmd = F_GETFD 或 F_GETFD)
3.获取/设置文件状态标记。(cmd = F_GETFL 或 F_SETFL)
4.获取/设置异步I/O所有权。(cmd = F_GETOWN 或 F_SETOWN)
5.获取/设置记录锁。(cmd = F_GETLK,F_SETLK或F_SETLKW)
F_DUPFD:复制一个文件描述符,新文件描述符作为返回值。它是尚未打开的各描述符中大于或者等于第三个参数arg
值中各值的最小值。新的描述符有它自己的一套文件描述符标志,其中close_on_exec标志被清除。
调用
- fcntl(filedes, F_DUPFD, 0);
等同于调用
关于dup函数的可以参见http://blog.csdn.net/todd911/article/details/11617519。
实践:
- #include <stdio.h>
- #include <fcntl.h>
-
- int main(void){
- int fd1=-1,fd2=-1;
- int result = 0;
- if((fd1 = open("a.txt",O_RDWR))<0){
- perror("open");
- result = -1;
- goto FINALLY;
- }
- printf("fd1:%d\n",fd1);
- if((fd2 = fcntl(fd1,F_DUPFD,0))<0){
- perror("fcntl");
- result = -1;
- goto FINALLY;
- }
-
- printf("fd2:%d\n",fd2);
-
- FINALLY:
- if(fd1 >= 0){
- close(fd1);
- }
- if(fd2 >= 0){
- close(fd2);
- }
- return result;
- }
运行结果:
fd1:3
fd2:4
F_GETFD:返回文件描述符对应的文件描述符标志,当前只定义了一个文件描述符标志FD_CLOEXEC(close_on_exec)。
F_SETFD:设置文件的文件描述符标志,新标志按第三个参数设置。
实践(设置文件的FD_CLOEXEC标志):
- #include <stdio.h>
- #include <fcntl.h>
-
- int main(void){
- int result = 0;
- int fd = -1;
- int flag;
-
- if((fd=open("a.txt", O_RDWR))<0){
- perror("open");
- result = -1;
- goto FINALLY;
- }
-
- flag = fcntl(fd, F_GETFD);
- flag |= FD_CLOEXEC;
- if(fcntl(fd, F_SETFD, flag)<0){
- perror("fcntl");
- result = -1;
- goto FINALLY;
- }
-
-
- FINALLY:
- if(fd >= 0){
- close(fd);
- }
- return result;
- }
F_GETFL:获取文件的状态标志作为函数的返回值。这边状态标志就是open函数中的状态。不幸的是,三个访问标志位
(O_RDONLY,O_WRONLY和O_RDWR)并不各占一位(这3种标志的值分别是0,1,2,由于历史原因,这三种值是互
斥),因此首先必须用屏蔽字O_ACCMODE获取访问模式位,然后将结果与这三种值中的任一种做比较。
F_SETFL:将文件状态设置为第三个参数的值。可以更改的几个标志位是:O_APPEND,O_NONBLOCK,O_SYNC,
O_DSYNC,O_RSYNC,O_FSYNC,O_ASYNC。
实践:
- #include <stdio.h>
- #include <fcntl.h>
-
- int main(void){
- int result = 0;
- int fd = -1, val;
-
- if((fd = open("a.txt", O_RDWR|O_APPEND))<0){
- perror("open");
- result = -1;
- goto FINALLY;
- }
-
- if((val = fcntl(fd,F_GETFL))<0){
- perror("fcntl");
- result = -1;
- goto FINALLY;
- }
-
- 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("unknown access mode");
- break;
- }
-
- if(val & O_APPEND){
- printf(",append");
- }else if(val & O_NONBLOCK){
- printf(",nonblock");
- }
- printf("\n");
-
- FINALLY:
- if(fd >= 0){
- close(fd);
- }
- return result;
- }
运行结果:
read write,append
F_SETFL和F_SETFD的使用方法差不多,先F_GETFL文件状态,然后使用|=设置想要的状态,最后再F_SETFL设置回去。
F_GETOWN:取当前接受SIGIO和SIGURG信号的进程ID和进程组ID。这两种信号会在以后介绍异步IO时说明。
F_SETOWN:设置接受SIGIO和SIGURG信号的进程ID和进程组ID。正的arg指定一个进程ID,负的arg表示等于arg绝对值
的一个进程组ID。