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

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

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

 

//写进程

#include

#include

#include

#include

#include

#include

#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

#include

#include

#include

#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。

 


 

  • 你不知道的Linux有名管道(FIFO)的阻塞和非阻塞读写已关闭评论
    A+
发布日期:2019年07月14日  所属分类:物联网