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

时间:2021-11-19 15:11:55

有名管道的读写有阻塞和非阻塞两种,可以在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。