0

0

c++怎么操作Linux下的命名管道FIFO_c++ 进程间通信与读写阻塞机制【实战】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-02 13:47:02

|

653人浏览过

|

来源于php中文网

原创

open() 打开 FIFO 会卡住是因为 POSIX 规定其默认阻塞式行为:O_RDONLY 需等待写端打开,O_WRONLY 需等待读端打开;解决方法是加 O_NONBLOCK 标志或协调两端启动顺序。

c++怎么操作linux下的命名管道fifo_c++ 进程间通信与读写阻塞机制【实战】

open() 打开 FIFO 时为什么会卡住?

Linux 下 mkfifo() 创建的命名管道默认是阻塞式打开的:open("myfifo", O_RDONLY) 会一直挂起,直到有另一个进程以 O_WRONLY 打开它;反过来,open("myfifo", O_WRONLY) 也会阻塞,直到有读端打开。这不是 bug,而是 POSIX FIFO 的设计行为。

常见错误现象:单侧启动程序后直接卡在 open(),strace 显示 openat(AT_FDCWD, "myfifo", O_RDONLY) 没有返回。

  • 解决方法一(推荐):用 O_NONBLOCK 标志避免阻塞,但需自行处理 EAGAIN/EWOULDBLOCK
  • 解决方法二:确保读写两端进程启动顺序合理,或用信号/临时文件协调启动时机
  • 注意:O_NONBLOCKO_RDONLYO_WRONLY 行为不同——只读端加该标志后,即使没写端也能成功打开;只写端加该标志后,若无读端则 open() 直接失败并设 errno = ENXIO

read() 和 write() 在 FIFO 上的阻塞逻辑

FIFO 的读写阻塞和普通文件完全不同:它依赖“两端是否就绪”。一旦写端关闭,读端 read() 会返回 0(表示 EOF);如果读端提前关闭,写端 write() 会触发 SIGPIPE 或返回 -1 并设 errno = EPIPE

关键细节:

立即学习C++免费学习笔记(深入)”;

  • 写入数据量 ≤ PIPE_BUF(通常 4096 字节)是原子的;超过则可能被拆包,且不保证顺序(多写端时尤其危险)
  • 读端未打开时,带 O_NONBLOCKopen(O_WRONLY) 失败;但只要读端曾经打开过、哪怕已关闭,open(O_WRONLY) 仍可能成功(内核保留 FIFO 状态直到所有 fd 关闭)
  • 不要依赖 select()poll() 对 FIFO 做“可写”判断——它们对只写端的就绪判断不可靠,Linux 内核文档明确说明 POLLOUT 在 FIFO 上总是就绪(即使没读端),容易误判

一个安全的 C++ FIFO 读写封装示例

下面是一个最小可行的 RAII 封装,规避常见陷阱(如未检查 open() 返回值、忽略 ENXIO、不处理 SIGPIPE):

VIVA
VIVA

一个免费的AI创意视觉设计平台

下载
#include 
#include 
#include 
#include 
#include 

class FifoWriter { int fd = -1; public: explicit FifoWriter(const char* path) { // 非阻塞打开,避免死等 fd = open(path, O_WRONLY | ONONBLOCK); if (fd == -1) { if (errno == ENXIO) { throw std::runtime_error("FifoWriter: no reader opened yet"); } throw std::runtime_error(std::string("open failed: ") + strerror(errno)); } // 忽略 SIGPIPE,让 write 返回 EPIPE 而非崩溃 signal(SIGPIPE, SIG_IGN); }

~FifoWriter() {
    if (fd_ != -1) close(fd_);
}

ssize_t write(const void* buf, size_t len) {
    return ::write(fd_, buf, len);
}

};

// 读端类似,但 open(O_RDONLY | O_NONBLOCK) 总是成功,无需 ENXIO 检查

为什么 unlink() 后还能读写?FIFO 的生命周期真相

unlink("myfifo") 只删除目录项,不影响已打开的 fd。只要至少一个进程还持有该 FIFO 的读或写 fd,内核就会维持其存在。这和普通文件一致,但容易被误解为“删了就没了”。

实战中要注意:

  • 服务进程重启前应先 unlink() 旧 FIFO,否则 mkfifo() 会失败(errno = EEXIST
  • 不要在读写过程中反复 unlink() + mkfifo() —— 正在使用的 FIFO 不会被真正销毁,但新创建的同名 FIFO 是另一个 inode,老进程继续读写旧的,新进程拿到新的,导致通信错乱
  • 调试时用 lsof | grep myfifo 查看哪些进程还在占用 FIFO,比单纯看文件是否存在更可靠

FIFO 的复杂点不在语法,而在状态协同:它强制要求你思考“谁先启动”“谁先关闭”“错误如何传播”。漏掉任意一环,程序就可能静默卡死或崩溃。

相关文章

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
磁盘配额是什么
磁盘配额是什么

磁盘配额是计算机中指定磁盘的储存限制,就是管理员可以为用户所能使用的磁盘空间进行配额限制,每一用户只能使用最大配额范围内的磁盘空间。php中文网为大家提供各种磁盘配额相关的内容,教程,供大家免费下载安装。

1345

2023.06.21

如何安装LINUX
如何安装LINUX

本站专题提供如何安装LINUX的相关教程文章,还有相关的下载、课程,大家可以免费体验。

700

2023.06.29

linux find
linux find

find是linux命令,它将档案系统内符合 expression 的档案列出来。可以指要档案的名称、类别、时间、大小、权限等不同资讯的组合,只有完全相符的才会被列出来。find根据下列规则判断 path 和 expression,在命令列上第一个 - ( ) , ! 之前的部分为 path,之后的是 expression。还有指DOS 命令 find,Excel 函数 find等。本站专题提供linux find相关教程文章,还有相关

294

2023.06.30

linux修改文件名
linux修改文件名

本专题为大家提供linux修改文件名相关的文章,这些文章可以帮助用户快速轻松地完成文件名的修改工作,大家可以免费体验。

773

2023.07.05

linux系统安装教程
linux系统安装教程

linux系统是一种可以免费使用,自由传播,多用户、多任务、多线程、多CPU的操作系统。本专题提供linux系统安装教程相关的文章,大家可以免费体验。

571

2023.07.06

linux查看文件夹大小
linux查看文件夹大小

Linux是一种自由和开放源码的类Unix操作系统,存在着许多不同的Linux版本,但它们都使用了Linux内核。Linux可安装在各种计算机硬件设备中,比如手机、平板电脑、路由器、视频游戏控制台、台式计算机、大型机和超级计算机。linux怎么查看文件夹大小呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

544

2023.07.20

linux查看ip命令
linux查看ip命令

本专题为大家提供linux查看ip命令相关文章内容,感兴趣的朋友可以免费下载体验试试。

296

2023.07.20

linux查看cpu使用率
linux查看cpu使用率

在linux的系统维护中,可能需要经常查看cpu使用率,分析系统整体的运行情况。本专题为大家带来了linux查看cpu使用率的相关文章,感兴趣的朋友千万不要错过了。

374

2023.07.25

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 6.4万人学习

Git 教程
Git 教程

共21课时 | 2.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号