fcntl

每个文件描述符都有一个close-on-exec标志。默认情况下,这个标志最后一位被设置为 0。这个标志符的具体作用在于当开辟其他进程调用exec()族函数时,在调用exec函数之前为exec族函数释放对应的文件描述符。


我们来看一下具体的实例。这是两个程序:

[c-sharp]  view plain copy print ?
  1. //file:fcntl  
  2. int main()  
  3. {  
  4.     pid_t pid;  
  5.     fd = open("test.txt",O_RDWR|O_APPEND);  
  6.     if (fd == -1)  
  7.     ##printf("open err/n");  
  8.     printf("fd = %d",fd);  
  9.     printf("fork!/n");  
  10.     fcntl(fd, F_SETFD, 1);                      
  11.     char *s="ooooooooooooooooooo";  
  12.     pid = fork();  
  13.     if(pid == 0)  
  14.     execl("ass""./ass", &fd, NULL);  
  15.     wait(NULL);  
  16.     write(fd,s,strlen(s));  
  17.     close(fd);  
  18.     return 0;  
  19. }  
  20. //ass 源代码  
  21. int main(int argc, char *argv[])  
  22. {  
  23.     int fd;  
  24.     printf("argc = %d ",argc);  
  25.     fd = *argv[1];  
  26.     printf("fd = %d",fd);  
  27.     char *s = "zzzzzzzzzzzzzzzzzzz";  
  28.     write(fd, (void *)s, strlen(s));  
  29.     close(fd);  
  30.     return 0;  
  31. }  

PS:那个test.txt提前已经创建 为空文件~

fcntl(fd, F_SETFD, 1) 此句将fd的close-on-exec 标志设置为1,开启此标志。那么当子进程调用execl函数时,execl执行ass,ass是不能向fd内写入的,因为在调用execl函数之前系统已经讲子进程的此文件描述符关闭了。(attention:这里是子进程!)
但是如果将 fcntl(fd, F_SETFD, 1)改为fcntl(fd, F_SETFD, 0),或者直接将此句注释掉,那么,ass便可以向这个文件描述符中任意添写东西了~~

PS:如果将fcntl设置为开启,即设置为1,那么,此文件描述符依然是可以被主进程操作的。

下面将程序执行的结过给大家:
当执行此句fcntl(fd, F_SETFD, 1)
fd = 3fork!
argc = 2 fd = 3
test.txt中的内容为:
ooooooooooooooooooo

当将fcntl(fd, F_SETFD, 1)注释掉或者将 1改为 0时
结果:
fd = 3fork!
argc = 2 fd = 3lost
test.txt中的内容为:
zzzzzzzzzzzzzzzzzzoooooooooooooooooooooo

 

转载自:http://blog.163.com/tlost_heaven/blog/static/79246122200910239562451/

xxxxxxxxxxxxxxxxxxxxxxxxxxxxx我 是 分 割 线xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx

文件控制函数

         fcntl -- file control
LIBRARY
         Standard C Library (libc, -lc)
SYNOPSIS
         #include ;
         int fcntl(int fd, int cmd, ...);
[描述]
            Fcntl()针对(文件)描述符提供控制.参数fd 是被参数cmd操作(如下面的描述)的描述符.
            针对cmd的值,fcntl能够接受第三个参数int arg
fcntl函数有5种功能: 
1.复制一个现有的描述符(cmd=F_DUPFD). 
2.获得/设置文件描述符标记(cmd=F_GETFD或F_SETFD). 
3.获得/设置文件状态标记(cmd=F_GETFL或F_SETFL). 
4.获得/设置异步I/O所有权(cmd=F_GETOWN或F_SETOWN). 
5.获得/设置记录锁(cmd=F_GETLK,F_SETLK或F_SETLKW). 
     cmd值:
            F_DUPFD            返回一个如下描述的(文件)描述符:
                            o            最小的大于或等于arg的一个可用的描述符
                            o            与原始操作符一样的某对象的引用
                            o            如果对象是文件(file)的话,返回一个新的描述符,这个描述符与arg 共享相同的偏移量(offset)
                            o            相同的访问模式(读,写或读/写)
                            o            相同的文件状态标志(如:两个文件描述符共享相同的状态标志)
                            o            与新的文件描述符结合在一起的close-on-exec 标志被设置成交叉式访问execve(2)的系统调用
     
    
           
F_GETFD           
取得与文件描述符fd联合close-on-exec标志,类似FD_CLOEXEC.如果返回值和FD_CLOEXEC进行与运算结果是0的话,文件保
持交叉式访问exec(),否则如果通过exec运行的话,文件将被关闭(arg 被忽略)
     
    
            F_SETFD            设置close-on-exec 旗标。该旗标以参数arg 的FD_CLOEXEC位决定。     
   
            F_GETFL            取得fd的文件状态标志,如同下面的描述一样(arg被忽略)
     
   
            F_SETFL            设置给arg描述符状态标志,可以更改的几个标志是: O_APPEND, O_NONBLOCK,O_SYNC和O_ASYNC。
     
     
            F_GETOWN             取得当前正在接收SIGIO或者SIGURG信号的进程id或进程组id,进程组id返回成负值(arg被忽略)
     
   
            F_SETOWN            设置将接收SIGIO和SIGURG信号的进程id或进程组id,进程组id通过提供负值的arg来说明,否则,arg将被认为是进程id。(可以这样理解:设置当这个文件fd可以读或者可以写的时候,将发送SIGIO和SIGURG信号给哪个进程ID或者进程组ID,)
   
            命令字(cmd)F_GETFL和F_SETFL的标志如下面的描述:
            O_NONBLOCK            非阻塞I/O;如果read(2)调用没有可读取的数据,或者如果write(2)操作将阻塞,read或write调用返回-1和EAGAIN错误
            O_APPEND                    强制每次写(write)操作都添加在文件大的末尾,相当于open(2)的O_APPEND标志
           
O_DIRECT                   
最小化或去掉reading和writing的缓存影响.系统将企图避免缓存你的读或写的数据.如果不能够避免缓存,那么它将最小化已经被缓存了的数据造
成的影响.如果这个标志用的不够好,将大大的降低性能
            O_ASYNC                    当I/O可用的时候,允许SIGIO信号发送到进程组,例如:当有数据可以读的时候
在修改文件描述符标志或文件状态标志时必须谨慎,先要取得现在的标志值,然后按照希望修改它,最后设置新标志值。不能只是执行F_SETFD或F_SETFL命令,这样会关闭以前设置的标志位。
fcntl的返回值 与命令有关。如果出错,所有命令都返回-1,如果成功则返回某个其他值。下列三个命令有特定返回值:F_DUPFD,F_GETFD,F_GETFL以及F_GETOWN。第一个返回新
的文件描述符,第二个返回相应标志,最后一个返回一个正的进程ID或负的进程组ID。
控制fd的例程 如下:
#include 
#include 
#include 
#include 
using namespace std; 
int main(int argc,char* argv[]) 

int fd,var; 
// fd=open("new",O_RDWR); 
if (argc!=2) 

perror("--"); 
cout

if((var=fcntl(atoi(argv[1]), F_GETFL, 0))

/**三个存取方式标志(O_RDONLY,O_WRONLY,以及O_RDWR)并不各占1位。(这三种标志的值各是0、1和2,由于历
史原因。这三种值互斥—一个文件只能有这三种值之一。)因此首先必须用屏蔽字O_ACCMODE取得存取方式位,然后将结果与这三种值相比较。
****/
switch(var & O_ACCMODE)  

case O_RDONLY : cout
break; 
case O_WRONLY : cout
break; 
case O_RDWR : cout
break; 
default : break; 

if (val & O_APPEND) 
cout
if (val & O_NONBLOCK) 
cout
cout
exit(0); 

. 获得/设置记录锁的功能: (cmd=F_GETLK,F_SETLK或F_SETLKW).  
          F_GETLK          通过第三个参数arg(一个指向flock的结构体)取得第一个阻塞lock
description指向的的锁.取得的信息将覆盖传到fcntl()的flock结构的信息.如果没有发现能够阻止本次锁(flock)生成的锁,这
个结构将不被改变,除非锁的类型被设置成F_UNLCK.
     
          F_SETLK         
按照指向结构体flock的指针的第三个参数arg所描述的锁的信息设置或者清除一个文件segment锁.F_SETLK被用来实现共享(或读)锁
(F_RDLCK)或独占(写)锁(F_WRLCK),同样可以去掉这两种锁(F_UNLCK).如果共享锁或独占锁不能被设置,fcntl()将立即返
回EAGAIN.
     
          F_SETLKW         
除了共享锁或独占锁被其他的锁阻塞这种情况外,这个命令和F_SETLK是一样的.如果共享锁或独占锁被其他的锁阻塞,进程将等待直到这个请求能够完成.
当fcntl()正在等待文件的某个区域的时候捕捉到一个信号,如果这个信号没有被指定SA_RESTART,fcntl将被中断.
     
          当一个共享锁被set到一个文件的某段的时候,其他的进程可以set 共享锁到这个段或这个段的一部分.共享所阻止任何其他进程set独占锁到这段保护区域的任何部分.如果文件描述符没有以读的访问方式打开的话,共享锁的设置请求会失败
    
          独占锁阻止任何其他的进程在这段保护区域任何位置设置共享锁或独占锁.如果文件描述符不是以写的访问方式打开的话,独占锁的请求会失败
结构体flock的指针
struct flcok 

short int l_type; /* 锁定的状态*/ 
//这三个参数用于分段对文件加锁,若对整个文件加锁,则:l_whence=SEEK_SET,l_start=0,l_len=0;
short int l_whence;/*决定l_start位置*/ 
off_t l_start; /*锁定区域的开头位置*/ 
off_t l_len; /*锁定区域的大小*/ 
pid_t l_pid; /*锁定动作的进程*/ 
};
l_type 有三种状态: 
F_RDLCK 建立一个供读取用的锁定 
F_WRLCK 建立一个供写入用的锁定 
F_UNLCK 删除之前建立的锁定 
l_whence 也有三种方式: 
SEEK_SET 以文件开头为锁定的起始位置。 
SEEK_CUR 以目前文件读写位置为锁定的起始位置 
SEEK_END 以文件结尾为锁定的起始位置。 
  
返回值      成功则返回0,若有错误则返回-1,错误原因存于errno.
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值