新闻中心
Python与C程序间管道通信中的文件描述符继承问题及解决方案

本文深入探讨了在python父进程通过`os.execl()`启动c子进程并尝试进行管道通信时,由于python 3.4+中`os.pipe()`返回的文件描述符默认为不可继承,导致的“bad file descriptor”错误。文章详细解释了文件描述符继承机制,对比了python和c在这一行为上的差异,并提供了使用`os.set_inheritable()`函数显式设置文件描述符继承性的解决方案,确保进程间通信的顺畅进行。
在跨语言或跨进程通信(IPC)场景中,管道(pipe)是一种常用且高效的机制,尤其适用于父子进程间的数据交换。当一个Python程序作为父进程,通过os.
fork()创建子进程后,再使用os.execl()加载并执行一个外部的C程序作为子进程时,如果尝试通过管道进行通信,可能会遇到“Bad file descriptor”错误。这种错误通常意味着子进程无法访问父进程创建的某个文件描述符,即使在逻辑上它应该被继承。
理解文件描述符继承机制
文件描述符(File Descriptor, FD)是操作系统用来标识打开文件或I/O资源的整数。当一个进程通过fork()创建子进程时,子进程通常会继承父进程的所有打开的文件描述符。然而,当子进程随后调用exec系列函数(如execl())加载并执行一个全新的程序时,文件描述符的继承行为就变得至关重要。
在Linux/Unix系统中,文件描述符有一个“close-on-exec”标志。如果这个标志被设置,那么当进程执行exec系列函数时,对应的文件描述符会自动关闭。如果未设置,文件描述符则会保留并传递给新的程序。
Python的os.pipe()函数在Python 3.4版本之后引入了一个重要的行为变更:它返回的文件描述符默认是不可继承的(即设置了“close-on-exec”标志)。这意味着,当父进程调用os.pipe()创建管道,然后fork()一个子进程,接着子进程调用os.execl()加载新的程序时,这些管道的文件描述符(特别是需要传递给C子进程的写入端)在exec调用时会被关闭,导致C子进程无法使用它们。
相比之下,传统的C语言pipe()系统调用所创建的文件描述符默认是可继承的(即未设置“close-on-exec”标志)。这是Python和C在管道IPC中行为差异的根本原因。
错误的现象:Python父进程与C子进程的通信失败
考虑以下场景:一个Python父进程创建管道,然后fork()并execl()一个C子进程,意图让C子进程向管道写入数据,父进程从管道读取。
Python父进程代码 (存在问题):
import os
import sys
def main():
r, w = os.pipe() # r: read end, w: write end
pid = os.fork()
if pid == 0: # 子进程
os.close(r) # 子进程关闭读取端
print(f'Child process: write fd = {w}', file=sys.stderr)
# 将写入端的文件描述符作为参数传递给C程序
name = './c_child' # 假设已编译好的C程序路径
# 在exec前,w是不可继承的,exec后w将被关闭
os.execl(name, name, str(w))
# 如果execl失败,下面的代码才会被执行
sys.exit(1)
else: # 父进程
os.close(w) # 父进程关闭写入端
os.waitpid(-1, 0) # 等待子进程结束
print('Parent receive: ', os.read(r, 10))
os.close(r)
if __name__ == "__main__":
main()C子进程代码 (用于接收Python父进程传递的fd):
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h> // For write, close
int main(int argc, char *argv[]) {
if (argc < 2) {
fprintf(stderr, "Usage: %s <file_descriptor>\n", argv[0]);
exit(EXIT_FAILURE);
}
char buf[] = "Hello Pipe!";
// 将字符串参数转换为整数文件描述符
int fd = (int)strtol(argv[1], NULL, 10);
fprintf(stderr, "C Child process: received fd = %d\n", fd);
// 尝试向该文件描述符写入
ssize_t count = write(fd, buf, sizeof(buf));
if (count == -1) {
perror("write error in C child"); // 这里会输出 "Bad file descriptor"
exit(EXIT_FAILURE);
} else {
printf("Child sent: %s\n", buf);
}
close(fd); // 关闭文件描述符
exit(EXIT_SUCCESS);
}编译C程序:gcc c_child.c -o c_child
DeepBrain
AI视频生成工具,ChatGPT +生成式视频AI =你可以制作伟大的视频!
146
查看详情
执行上述Python代码,会观察到类似如下的输出:
Child process: write fd = 4 C Child process: received fd = 4 write error in C child: Bad file descriptor Parent receive: b''
这明确表明,C子进程收到了文件描述符的数值(例如4),但在尝试写入时,该描述符已经失效,从而报告了“Bad file descriptor”错误。父进程也因此无法从管道中读取到任何数据。
解决方案:显式设置文件描述符继承性
为了解决这个问题,我们需要在Python父进程中,显式地将管道的写入端文件描述符设置为可继承,然后再执行execl()。这可以通过os.set_inheritable()函数实现。
os.set_inheritable(fd, inheritable)函数用于设置或清除文件描述符fd的“close-on-exec”标志。当inheritable为True时,文件描述符将变为可继承;当为False时,则变为不可继承。
Python父进程代码 (修正后):
import os
import sys
def main():
r, w = os.pipe() # r: read end, w: write end
# 关键一步:将写入端文件描述符设置为可继承
os.set_inheritable(w, True)
pid = os.fork()
if pid == 0: # 子进程
os.close(r) # 子进程关闭读取端
print(f'Child process: write fd = {w}', file=sys.stderr)
name = './c_child' # 假设已编译好的C程序路径
os.execl(name, name, str(w))
sys.exit(1)
else: # 父进程
os.close(w) # 父进程关闭写入端
os.waitpid(-1, 0) # 等待子进程结束
received_data = os.read(r, 10)
print('Parent receive: ', received_data.decode().strip()) # 解码并打印
os.close(r)
if __name__ == "__main__":
main()使用修正后的Python代码,再次执行:
Child process: write fd = 4 C Child process: received fd = 4 Child sent: Hello Pipe! Parent receive: Hello Pipe!
此时,通信成功。C子进程能够顺利地向管道写入数据,父进程也能够读取到这些数据。
关键要点与最佳实践
- 文件描述符继承性: 在Python 3.4及更高版本中,os.pipe()创建的文件描述符默认是不可继承的。如果需要在os.execl()或subprocess.Popen等调用后,子进程依然能够访问这些文件描述符,必须使用os.set_inheritable(fd, True)显式设置。
-
关闭不必要的管道端: 无论是在父进程还是子进程中,始终要关闭那些不使用的管道端。
- 父进程通常只读取,应关闭写入端。
- 子进程如果只写入,应关闭读取端。
- 这不仅能避免资源泄露,还能防止潜在的死锁(例如,如果写入端未关闭,读取端可能会一直等待,即使没有更多数据要写入)。
- 错误处理: 始终包含适当的错误处理机制,例如检查os.fork()、os.pipe()和write()的返回值,并使用perror()或打印错误信息。
- 参数传递: 当通过命令行参数将文件描述符传递给子进程时,确保子进程能够正确地将字符串参数解析为整数文件描述符。
通过理解Python中文件描述符的默认继承行为,并采取适当的措施(如os.set_inheritable()),可以有效地解决Python父进程与外部C子进程通过管道进行IPC时遇到的“Bad file descriptor”问题,从而构建健壮可靠的跨语言通信系统。
以上就是Python与C程序间管道通信中的文件描述符继承问题及解决方案的详细内容,更多请关注其它相关文章!
# 是在
# 睢宁网站推广前景
# 短视频seo官方
# 提升网站关键词排名吗
# 济南专业的网站设计优化
# 遂平抖音seo关键词排名
# 莱芜营销推广需求
# 无锡市免费网站推广批发
# sem运营和seo
# 重庆网站制作和推广公司
# 高明百度关键词排名优化
# 你可以
# 是一种
# linux
# 这是
# 设置为
# 死锁
# 加载
# 信中
# 命令行
# python程序
# unix
# ai
# 操作系统
# c语言
# python
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
qq游戏手机版下载安装_qq游戏移动端入口
Pandas DataFrame 高效批量赋值:告别循环与笛卡尔积误区
印象笔记怎样用批量导出备知识库_印象笔记用批量导出备知识库【备份方法】
我的世界官方游戏入口 我的世界官网平台直达链接
期待已久:小米17 Ultra、小米首款NAS本月登场
快手官方唯一登录入口 谨防山寨钓鱼网站
小红书商家版怎样在笔记嵌入商品卡路径_小红书商家版在笔记嵌入商品卡路径【挂载教程】
HTML空白字符处理机制:渲染、DOM与编码实践
动漫花园资源网使用步骤_动漫花园资源网下载流程
谷歌邮箱网页版官方页面入口 谷歌邮箱网页端快速访问
微博网页版直接访问 微博网页版账号管理快速入口
C#如何安全地从用户上传的XML文件中读取数据? 验证与清理策略
TikTok评论显示延迟如何处理 TikTok评论刷新优化方法
蓝湖怎样用切图标注提对接效率_蓝湖用切图标注提对接效率【设计对接】
顺丰快件物流信息 官方网站查询入口
windows10怎么查看本机ip_windows10命令提示符ipconfig使用
Win11怎么关闭快速启动_Win11彻底关机设置教程
打开就能玩的植物大战僵尸 植物大战僵尸网页版传送门
晋江读书网页版在线登录 晋江读书电脑版官网
CSS如何设置hover状态颜色_hover伪类调整背景或文字颜色
Golang如何处理RPC请求负载均衡_Golang RPC请求负载均衡策略与实践
不会效仿卡普空!《铁拳》制作人澄清:不采取赛事付费|直播|
《马克思佩恩3》早期版本曝光 UI设计曾多次调整!
qq音乐在线播放入口_qq音乐电脑版登录链接
一加手机拍照效果不好怎么办 一加哈苏影像调校与专业模式使用教程【高手篇】
漫蛙2正版漫画站 漫蛙2网页版快速访问入口
C++如何检测键盘输入_C++ _kbhit与_getch函数非阻塞输入
韩小圈电脑版在线入口_网页版免费登录地址
荣耀Play7TPro怎样在信息App置顶客服对话_iPhone荣耀Play7TPro信息App置顶客服对话【优先查看】
J*a应用程序首次运行自动创建文件与目录的最佳实践
Steam官网入口直达 Steam注册及登录步骤
mysql通配符支持数字匹配吗_mysql通配符能否用于数字匹配的解析
mcjs网页版在线存档 mcjs云存档登录入口
拼多多购物车商品数量无法修改如何处理 拼多多购物车操作优化方法
ArrayList与LinkedList操作复杂度详解:遍历与修改
UC浏览器如何安装插件 UC浏览器添加扩展程序详细教程【进阶】
抖音网页版平台入口 抖音网页版官网在线访问教程
星露谷物语官网入口 星露谷物语游戏官网入口
如何在Promise链中优雅地中断后续then执行
word中如何让数字纵向排列_Word数字纵向排列方法
新手怎么开始学化妆 零基础化妆入门教程
Animex动漫社网入口地址 Animex动漫社网正版在线入口
AO3镜像入口大全 AO3网页版内容访问全集
Golang切片为何属于引用类型_Golang slice底层结构与引用语义说明
Promise错误处理:在catch后终止链式then执行的策略
在哪找SublimeJ远程工具_SFTP插件配置教程
知乎APP怎么管理已购盐选内容_知乎APP盐选内容购买记录与查看方法
微信商城在哪里打开【步骤】
内存检查:在VS Code中调试C++时的内存视图
在Socket.IO连接中实现Access Token自动更新与动态重连


2025-12-13
浏览次数:次
返回列表