Linux有名管道(FIFO)的阻塞和非阻塞读写

有名管道读写实验

有名管道的读写有阻塞和非阻塞两种,可以在open()时指定,下面我们对各种情况进行一些讨论。

//写进程
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <fcntl.h>
#include <string.h>
#include <stdio.h>
#define FIFO_NAME "/tmp/myfifo"
main()
{
    int fd;
    char w_buf[50];
    int w_num;

    // 若fifo已存在,则直接使用,否则创建它
    if((mkfifo(FIFO_NAME,0777)<0)&&(errno!=EEXIST))
    {
        printf("cannot create fifo...\n");
        exit(1);
    }

    //以阻塞型只写方式打开fifo
    fd=open(FIFO_NAME,O_WRONLY);
    w_num=write(fd,"abcdg\0",6);
    printf("%d\n",w_num);
}


//读进程
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <fcntl.h>
#define FIFO_NAME "/tmp/myfifo"
main()
{
    char r_buf[50];
    int  fd;
    int  r_num;

// 若fifo已存在,则直接使用,否则创建它
    if((mkfifo(FIFO_NAME,0777)<0)&&(errno!=EEXIST))
    {
        printf("cannot create fifo...\n");
        exit(1);
    }
    //以阻塞型只读方式打开fifo
    fd=open(FIFO_NAME,O_RDONLY);
    if(fd==-1)
    {
        printf("open %s for read error\n");
        exit(1);
    }

    // 通过键盘输入字符串,再将其写入fifo,直到输入"exit"为止
    r_num=read(fd,r_buf,6);
    printf(" %d bytes read:%s\n",r_num,r_buf);

    unlink(FIFO_NAME);//删除fifo
}




1.

写进程阻塞,读进程阻塞

先运行写进程(被阻塞),再运行读进程,一切正常。
先运行读进程(被阻塞)运行写进程一切正常。

2.

写进程阻塞,读进程非阻塞

就改一句代码 fd=open(FIFO_NAME,O_RDONLY | O_NONBLOCK),下面类似。

先运行写进程(被阻塞),再运行读进程,一切正常。

先运行读进程,程序直接崩掉(Segmentation fault (core dumped)),想想也挺自然的,没东西你还要读,而且不愿等。。。

3.

写进程非阻塞,读进程阻塞

先运行写进程,open调用将返回-1,打开失败。

先运行读进程(被阻塞)再运行写进程一切正常。


4.

写进程非阻塞,读进程非阻塞

其实就是上面2,3类各取一半不正常的情况。。

还有我们在/tmp目录下通过ls -la命令可以看到管道文件myfifo的大小总是0,这是因为虽然FIFO文件存在于文件系统中,但FIFO中的内容都存放在内存中,所以文件大小始终为0。

<think>好的,用户想了解Linux有名管道FIFO)的循环使用方法示例代码,以及解决方案。首先,我需要回忆有名管道的基本概念。FIFO是一种特殊的文件,允许无亲缘关系的进程间通信。用户提到“循环操作”,可能是指如何持续读写FIFO,或者在多线程、循环结构中正确使用它。 接下来,我需要考虑FIFO的常见问题,比如阻塞非阻塞模式。默认情况下,打开FIFO阻塞直到另一端也打开,这可能影响循环操作。用户可能需要知道如何设置非阻塞模式,或者如何处理读写中的阻塞情况。 然后,结合用户提供的引用内容,引用[2]提到了使用互斥锁保护FIFO操作,这适用于多线程环境。所以,在示例代码中可能需要包含线程安全同步机制,比如pthread_mutex_lock/unlock。此外,用户可能需要循环读取或写入数据,比如服务器持续监听FIFO请求,或者客户端不断发送数据。 另外,用户提到“解决循环问题”,可能指FIFO在循环使用中的常见错误,比如读写端关闭后重新打开的问题,或者如何保持FIFO的持久连接。需要解释如何正确打开关闭文件描述符,以及处理EOF的情况。例如,当写端关闭时,读端会读到EOF,这时候可能需要重新打开FIFO才能继续读取。 然后,我需要组织一个示例代码,展示服务器客户端如何使用FIFO进行循环通信。服务器端可能在一个循环中读取请求并处理,客户端则循环发送数据。要注意打开模式,比如非阻塞标志O_NONBLOCK的使用,以及错误处理。 最后,总结常见问题,比如同步机制、阻塞处理、资源释放等,确保用户能够理解并避免常见陷阱。同时,生成相关问题,帮助用户进一步学习,比如多进程使用FIFO或与非阻塞IO结合的方法。</think>### Linux有名管道(FIFO)循环使用指南 #### 一、FIFO循环操作原理 1. FIFO通过`mkfifo`命令创建后,表现为特殊类型的文件,支持多进程/线程的循环读写操作 2. 保持读写端持续打开是实现循环通信的关键,需要避免过早关闭文件描述符[^1] 3. 推荐使用非阻塞模式(`O_NONBLOCK`)配合轮询机制实现稳定循环操作[^2] #### 二、示例代码实现 **服务端持续监听(server.c)**: ```c #include <fcntl.h> #include <sys/stat.h> #include <unistd.h> #define FIFO_PATH "/tmp/myfifo" int main() { mkfifo(FIFO_PATH, 0666); int fd = open(FIFO_PATH, O_RDONLY | O_NONBLOCK); char buffer[1024]; while(1) { ssize_t bytes_read = read(fd, buffer, sizeof(buffer)); if(bytes_read > 0) { // 处理接收到的数据 write(STDOUT_FILENO, buffer, bytes_read); } usleep(100000); // 100ms间隔轮询 } close(fd); return 0; } ``` **客户端持续写入(client.c)**: ```c #include <fcntl.h> #include <unistd.h> #define FIFO_PATH "/tmp/myfifo" int main() { int fd = open(FIFO_PATH, O_WRONLY); for(int i=0; i<10; i++) { char msg[] = "Message "; write(fd, msg, sizeof(msg)); sleep(1); } close(fd); return 0; } ``` #### 三、关键问题解决方案 1. **循环中断问题**: - 使用`fcntl(fd, F_SETFL, O_NONBLOCK)`设置非阻塞模式 - 配合`select()`/`poll()`实现高效事件驱动模型 ```c fd_set readfds; FD_ZERO(&readfds); FD_SET(fd, &readfds); select(fd+1, &readfds, NULL, NULL, NULL); ``` 2. **线程安全方案**: ```c pthread_mutex_t fifo_mutex = PTHREAD_MUTEX_INITIALIZER; void* thread_func(void* arg) { pthread_mutex_lock(&fifo_mutex); // FIFO操作代码 pthread_mutex_unlock(&fifo_mutex); return NULL; } ``` 3. **EOF处理策略**: ```c if(bytes_read == 0) { close(fd); fd = open(FIFO_PATH, O_RDONLY | O_NONBLOCK); // 重新打开 } ``` #### 四、最佳实践建议 1. 每次通信后保留至少一个打开的文件描述符 2. 使用`SIGPIPE`信号处理避免写入已关闭的FIFO 3. 定期检查FIFO状态:`fstat(fd, &st)`
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值