执行时关闭标识位 FD_CLOEXEC 的作用
首先先回顾 apue 中对它的描述:
① 表示描述符在通过一个 exec 时仍保持有效(书P63,3.14节 fcntl 函数,在讲 F_DUPFD 时顺便提到)
② 对打开文件的处理与每个描述符的执行时关闭(close-on-exec)标志值有关。
见图 3-1 节中对 FD_CLOEXEC 的说明,进程中每个打开描述符都有一个执行时关闭标志。若此标志设置,
则在执行 exec 时关闭该描述符,否则该描述符仍打开。除非特地用 fcntl 设置了该标志,否则系统的默认
操作是在执行 exec 后仍保持这种描述符打开。(书P190,8.10节 exec 函数)
概括为:
① FD_CLOEXEC 是“文件描述符”的标志
② 此标志用来控制在执行 exec 后,是否关闭对应的文件描述符
(关闭文件描述符即不能对文件描述符指向的文件进行任何操作)
下面以一个例子进行说明,包含两个独立程序,一个用来表示父进程,另一个表示它的子进程
父进程 parent.c:
// parent.c
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h> int main()
{
int fd = open("test.txt",O_RDWR|O_APPEND); if (fd == -)
{
printf("The file test.txt open failed ! The fd = %d\n",fd);
execl( "/bin/touch", "touch", "test.txt", (char*)NULL );
return ;
}
else
{
printf("The file test.txt open success ! The fd = %d\n", fd);
} printf("fork!\n"); // 什么也不写,相当于系统默认 fcntl(fd, F_SETFD, 0) ,即用 execl 执行子进程时,
// 不打开“执行时关闭”标识位 FD_CLOEXEC,此时子进程可以向 test.txt 写入字符串
char *s="The Parent Process Writed !\n"; pid_t pid = fork();
if(pid == ) /* Child Process */
{
printf("***** exec child *****\n");
execl("child", "./child", &fd, NULL);
printf("**********************\n");
} // 等待子进程执行完毕
wait(NULL);
ssize_t writebytes = write(fd,s,strlen(s));
if ( writebytes == - )
{
printf("The Parent Process Write To fd : %d Failed !\n", fd);
} close(fd);
return ;
}
子进程 child.c
//child.c
#include <stdio.h>
#include <unistd.h>
#include <string.h> int main(int argc, char *argv[])
{
printf("argc = %d\n",argc); if ( argv[] == NULL )
{
printf("There is no Parameter !\n");
return ;
} int fd = *argv[];
printf("child fd = %d\n",fd); char *s = "The Child Process Writed !\n";
ssize_t writebytes = write(fd, (void *)s, strlen(s));
if ( writebytes == - )
{
printf("The Child Process Write To fd : %d Failed !\n", fd);
} close(fd);
return ;
}
此时观察 test.txt ,得到结果
The Child Process Writed !
The Parent Process Writed !
因为代码中没做任何操作,系统默认是不设置“执行时关闭标识位”的。
现在在代码中进行设置这个标志:
printf("fork!\n");
fcntl(fd, F_SETFD, 1);
char *s="The Parent Process Writed !\n";
…………后面代码省略
此时再观察 test.txt,发现只能看到父进程的输出了:
The Parent Process Writed !
更标准的写法是:
…………前面代码省略
printf("fork!\n");
// 和 fcntl(fd, F_SETFD, 1) 等效,但这是标准写法,即用 FD_CLOEXEC 取代直接写1
int tFlags = fcntl(fd, F_GETFD);
fcntl(fd, F_SETFD, tFlags | FD_CLOEXEC ); char *s="The Parent Process Writed !\n";
…………后面代码省略
推荐后面一种写法。
如果在后面重新进行设置 fcntl(fd, F_SETFD, 0) ,即可重新看到子进程的输出(读者可以自己尝试)。
那么问题来了,如果子进程不使用 exec 函数执行的这种方式呢?
那么理论上设置这个标志是无效的。
// parent.c
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h> int main()
{
int fd = open("test.txt",O_RDWR|O_APPEND); if (fd == -)
{
printf("The file test.txt open failed ! The fd = %d\n",fd);
execl( "/bin/touch", "touch", "test.txt", (char*)NULL );
return ;
}
else
{
printf("The file test.txt open success ! The fd = %d\n", fd);
} printf("fork!\n"); // 系统默认 fcntl(fd, F_SETFD, 0) ,即用 execl 执行子进程时,
// 不打开“执行时关闭”标识位 FD_CLOEXEC
//fcntl(fd, F_SETFD, 1);
//fcntl(fd, F_SETFD, 0); // 和 fcntl(fd, F_SETFD, 1) 等效,但这是标准写法,即用 FD_CLOEXEC 取代直接写1
int tFlags = fcntl(fd, F_GETFD);
fcntl(fd, F_SETFD, tFlags | FD_CLOEXEC ); char *s="The Parent Process Writed !\n"; pid_t pid = fork();
if(pid == ) /* Child Process */
{
printf("***** exec child *****\n"); // execl("child", "./child", &fd, NULL);
// 注意下面,子进程不用 exec 函数,而是改成直接写入处理
// 此时文件描述符标识位 FD_CLOEXEC 不再起作用
// 即使设置这个标识位,子进程一样可以写入
char *s = "The Child Process Writed !\n";
ssize_t writebytes = write(fd, (void *)s, strlen(s));
if ( writebytes == - )
{
printf("Child Process Write To fd : %d Failed !\n", fd);
} printf("**********************\n");
// 注意这里结束子进程,但不要关闭文件描述符,否则父进程无法写入
exit();
} // 等待子进程执行完毕
wait(NULL);
ssize_t writebytes = write(fd,s,strlen(s));
if ( writebytes == - )
{
printf("The Parent Process Write To fd : %d Failed !\n", fd);
} close(fd);
return ;
}
注意修改后的地方:
if(pid == ) /* Child Process */
{
printf("***** exec child *****\n"); // execl("child", "./child", &fd, NULL);
// 注意下面,子进程不用 exec 函数,而是改成直接写入处理
// 此时文件描述符标识位 FD_CLOEXEC 不再起作用
// 即使设置这个标识位,子进程一样可以写入
char *s = "The Child Process Writed !\n";
ssize_t writebytes = write(fd, (void *)s, strlen(s));
if ( writebytes == - )
{
printf("The Child Process Write To fd : %d Failed !\n", fd);
} printf("**********************\n");
// 注意这里结束子进程,但不要关闭文件描述符,否则父进程无法写入
exit();
}
在前面仍然要设置标志:
int tFlags = fcntl(fd, F_GETFD);
fcntl(fd, F_SETFD, tFlags | FD_CLOEXEC );
重新编译,观察结果,发现子进程又可以重新写文件了:
The Child Process Writed !
The Parent Process Writed !
证明设置这个标志,对不用 exec 的子进程是没有影响的。
执行时关闭标识位 FD_CLOEXEC 的作用的更多相关文章
- 理清PHP在Linxu下执行时的文件权限
首先推荐一个linux权限的视频:Linux权限管理之基本权限,讲的非常好,看完后就基本明白了. 一.文件权限及所属 1.文件有三种类型的权限,为了方便期间,可以用数字来代替,这样可以通过数字的加减, ...
- IOS -执行时 (消息传递 )
一 函数调用概述 Objective-C不支持多重继承(同Java和Smalltalk),而C++语言支持多重继承. Objective-C是动态绑定,它的类库比C++要easy操作. Ob ...
- ZT fcntl设置FD_CLOEXEC标志作用
fcntl设置FD_CLOEXEC标志作用 分类: C/C++ linux 2011-11-02 22:11 3217人阅读 评论(0) 收藏 举报 bufferexegccnullfile 通过fc ...
- java内存结构(执行时数据区域)
java虚拟机规范规定的java虚拟机内存事实上就是java虚拟机执行时数据区,其架构例如以下: 当中方法区和堆是由全部线程共享的数据区. Java虚拟机栈.本地方法栈和程序计数器是线程隔离的数据区. ...
- SQLSERVER:大容量导入数据时保留标识值 (SQL Server)
从MSDN上看到实现大容量导入数据时保留标识值得方法包含三种: MSDN链接地址为:https://msdn.microsoft.com/zh-cn/library/ms178129.aspx 感觉M ...
- iOS执行时工具-cycript
cycript是大神saurik开发的一个很强大的工具,能够让开发人员在命令行下和应用交互,在执行时查看和改动应用.它确实能够帮助你破解一些应用,但我认为这个工具主要还是用来学习其它应用的设计(主要是 ...
- CentOS安装redis-audit 但执行时出错未解决 记录一下安装过程
网上很多安装过程都太老了,测试很多方法终于成功了,但执行时还是出错,哪位熟悉的可以告知一下. yum install -y ruby rubygems ruby-devel git gcc gem s ...
- VC++ MFC单文档应用程序SDI下调用glGenBuffersARB(1, &pbo)方法编译通过但执行时出错原因分析及解决办法:glewInit()初始化的错误
1.问题症状 在VC++环境下,利用MFC单文档应用程序SDI下开发OpenGL程序,当调用glGenBuffersARB(1, &pbo)方法编译通过但执行时出错,出错代码如下: OpenG ...
- VS2015 调试中断点突然失效的解决办法、VS调试时关闭调试让浏览器继续保留页面
VS2010 调试中断点突然失效的解决办法 问题描述:在调试前加了断点,但debug时红色的断点变成透明的圆圈加一个感叹号,执行到该处时也不会停止. 这个问题遇到过几次了,前几次都没怎么注意,有时候是 ...
随机推荐
- (原创)Windows系统后安装ubuntu,无法选择启动ubuntu。
继Window系统之后,安装Ubuntu系统. 问题:启动没有Grub的ubuntu启动项. 查看:/boot/grub/中只有txt和env,内容空白,grub没有设置好. 修复: sudo fdi ...
- JavaScript中的防篡改对象
由于JavaScript共享的特性,任何对象都可以被放在同一环境下运行的代码修改. 例如: var person = {name:"caibin'} person.age = 21; 即使第 ...
- PBOC~PPT-补充内容B(转)
PBOC电子现金基于借记/贷记应用上小额支付的一种实现.在借记卡上可以解释为预付,在贷记卡上可以解释为预先授权.预付的金额或预授权额度在卡片中体现为可脱机消费的金额,也就是电子现金余额.电子现金解决方 ...
- PHP Cookie Session
这些都是基础知识,不过有必要做深入了解.先简单介绍一下. 二者的定义: 当你在浏览网站的时候,WEB 服务器会先送一小小资料放在你的计算机上,Cookie 会帮你在网站上所打的文字或是一些选择, 都纪 ...
- python数据类型
整数 十六进制需要用前缀0x+0-9+a-f表示,如0xff00 浮点数(小数) 可以用1.23,1.56e9(科学计数法)表示 字符串 ""或''括起来都行 布尔值 只有True ...
- latex 竖排子图的生成
latex命令如下: 需要的包为: \usepackage{graphicx} \usepackage{subfigure} \begin{figure*}%加*的作用是跨栏(双栏和单栏latex的区 ...
- 真核转录组(denovo/resequencing)及案例分析
参考: 转录组文章的常规套路 文章解读:<Science>小麦转录组研究文章 转录组数据饱和度评估方法 Paper这个东西是多么的诱人,可以毕业,可以评职称,可以拿绩效. 现在的文章都是有 ...
- SSIS数据转换后数值总数差异过大
之前做过一个项目,犯了一个小错误,写出来给大家分享一下,以防大家出同样的错误. 做了一个ETL包,对货品的销售额进行数据转换,字符型 --〉 浮点型: 之后对销售额进行求和,在测试数据结果时发现与销售 ...
- linux限制ftp账户的访问路径
1.建用户,命令行状态下,在root用户下: 运行命令:"useradd -d /home/test test" //增加用户test,并制定test用户的主目录为/home/te ...
- javascript的假查询
1. function select(){ var text=$("#ss").val();//获得关键字 $("#show_tab tr").hide().f ...