网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
一、fcntl函数
fcntl函数可以改变或者查看已打开文件的性质。该函数的定义如下:
#include <fcntl.h>
int fcntl(int fd, int cmd, ... /* arg */ );
fcntl有5种功能:
- 复制一个现有的描述符。(cmd = F_DUPFD)
- 获取/设置文件描述符标记。(cmd = F_GETFD 或 F_GETFD)
- 获取/设置文件状态标记。(cmd = F_GETFL 或 F_SETFL)
- 获取/设置异步I/O所有权。(cmd = F_GETOWN 或 F_SETOWN)
- 获取/设置记录锁。(cmd = F_GETLK、F_SETLK或F_SETLKW)
我们这里重点分析10种cmd值中的前7种:
F_DUPFD:复制一个文件描述符,新文件描述符作为返回值。它是尚未打开的各描述符中大于或者等于第三个参数arg
值中各值的最小值。新描述符与filedes共享同一文件表项。但是新的描述符有它自己的一套文件描述符标志,其中close_on_exec标志被清除。
fcntl(filedes, F\_DUPFD, 0);
//等同于调用
dup(filedes);
F_GETFD:对应于filedes的文件描述符标志作为函数值返回。
F_SETFD:对应于filedes设置文件描述符标志,新标志按第三个参数(取为整数值)设置。
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。
F_GETOWN:取当前接收SIGIO和SIGURG信号的进程ID和进程组ID。
F_SETOWN:设置接受SIGIO和SIGURG信号的进程ID和进程组ID。正的arg指定一个进程ID,负的arg表示等于arg绝对值的一个进程组ID。
这里首先用《UNP》中一个例子来演示:打印指定描述符的文件标志
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <fcntl.h>
int main()
{
int fd;
int var;
if ((fd = open("test.txt", O_RDWR | O_CREAT | O_EXCL, 0664)) == -1)
{
perror("open error");
exit(1);
}
if ((var = fcntl(fd, F_GETFL, 0)) < 0)
{
perror("fcntl error");
close(fd);
exit(1);
}
switch(var & O_ACCMODE)
{
case O_RDONLY:
printf("Read only..\n");
break;
case O_WRONLY:
printf("Write only..\n");
break;
case O_RDWR:
printf("Read And Write..\n");
break;
default:
printf("Do't know..\n");
break;
}
if (var & O_APPEND)
{
printf("And Append..\n");
}
if (var & O_NONBLOCK)
{
printf("And Nonblocking..\n");
}
if (close(fd) == -1)
{
perror("close error");
}
return 0;
}
运行:
当然,更常见的用法是:在修改文件描述符标志或文件状态标志时,先取得现有的状态值,然后根据需要修改它,最后设置新标志值。
示例:
增加文件的某个flags,比如文件是阻塞的,想设置成非阻塞:
flags = fcntl(fd,F_GETFL,0);
flags |= O_NONBLOCK;
fcntl(fd,F_SETFL,flags);
取消文件的某个flags,比如文件是非阻塞的,想设置成为阻塞:
flags = fcntl(fd,F_GETFL,0);
flags &= ~O_NONBLOCK;
fcntl(fd,F_SETFL,flags);
unp中将其封装为函数:
void set_fl(int fd, int flags)
{
int val;
if( (val = fcntl(fd,F_GETFL,0)) < 0)
perror("fcntl F\_GETFL error");
val |= flags; //或运算,添加标志位
//val &= ~flags; //与运算,清除标志位
if( fcntl(fd,F_SETFL, val) <0)
perrror("fcntl F\_SETFL error");
}
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
究,那么很难做到真正的技术提升。**
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!