查看: 90|回覆: 1

[教程] linux 命名管道实例详解

[複製鏈接]

3

主題

0

回帖

0

積分

热心网友

金币
0
閲讀權限
220
精華
0
威望
0
贡献
0
在線時間
0 小時
註冊時間
2010-8-1
發表於 2023-9-21 00:00:00 | 顯示全部樓層 |閲讀模式

linux进程间通信——命名管道

  fifo(命名管道)不同于匿名管道之处在于它提供⼀个路径名与之关联,以fifo的⽂件形式存储于⽂件系统中。命名管道是⼀个设备⽂件,因此,即使进程与创建fifo的进程不存在亲缘关系,只要可以访问该路径,就能够通过fifo相互通信。值得注意的是,fifo(first input first output)总是按照先进先出的原则⼯作,第⼀个被写⼊的数据将⾸先从管道中读出。

  创建命名管道的系统函数有两个:mknod和mkfifo。两个函数均定义在头⽂件sys/stat.h,函数原型如下:

?
1
2
3
4
#include <sys/types.h>
#include <sys/stat.h>
int mknod(const char *path,mode_t mod,dev_t dev);
int mkfifo(const char *path,mode_t mode);

   函数mknod参数中path为创建的命名管道的全路径名:mod为创建的命名管道的模式,指明其存取权限;dev为设备值,该值取决于⽂件创建的种类,它只在创建设备⽂件时才会⽤到。这两个函数调⽤成功都返回0,失败都返回-1。下⾯使⽤mknod函数创建了⼀个命名管道:

?
1
2
3
4
5
6
7
8
9
10
11
umask(0);
 
if (mknod("/tmp/fifo",s_ififo | 0666) == -1)
 
{
 
perror("mkfifo error");
 
exit(1);
 
}

 函数mkfifo前两个参数的含义和mknod相同。下⾯是使⽤mkfifo的⽰例代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
umask(0);
 
if (mkfifo("/tmp/fifo",s_ififo|0666) == -1)
 
{
 
 
perror("mkfifo error!");
 
exit(1);
 
}

下面为一个试例:

read端

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
#include<stdlib.h>
#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<errno.h>
#define path "./fifo"
#define size 128
int main()
{
 umask(0);
 if (mkfifo (path,0666|s_ififo) == -1)
 {
 perror ("mkefifo error");
 exit(0);
 }
 int fd = open (path,o_rdonly);
 if (fd<0)
 {
  printf("open fd is error\n");
  return 0;
 }
 
 char buf[size];
 while(1){
 ssize_t s = read(fd,buf,sizeof(buf));
 if (s<0)
 {
  perror("read error");
  exit(1);
 }
 else if (s == 0)
 {
  printf("client quit! i shoud quit!\n");
  break;
 }
 else
 {
  buf = '\0';
  printf("client# %s ",buf);
  fflush(stdout);
 }
 }
 close (fd);
 return 3;
}

下面为weite端:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
#include<stdlib.h>
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<string.h>
#include<errno.h>
#include<fcntl.h>
 
#define path "./fifo"
#define size 128
int main()
{
 int fd = open(path,o_wronly);
 if (fd < 0)
 {
  perror("open error");
  exit(0);
 }
 
 char buf[size];
 while(1)
 {
  printf("please enter#:");
  fflush(stdout);
  ssize_t s = read(0,buf,sizeof(buf));
  if (s<0)
  {
   perror("read is failed");
   exit(1);
  }
  else if(s==0)
  {
   printf("read is closed!");
   return 1;
  }
  else{
   buf= '\0';
   write(fd,buf,strlen(buf));
  }
 }
 return 0;
}

打开两个终端:

 

linux 命名管道实例详解
linux 命名管道实例详解

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

原文链接:http://blog.csdn.net/qq_35116353/article/details/59117339

回覆

使用道具 舉報

0

主題

720

回帖

4441

積分

琼殿精英

金币
3721
閲讀權限
220
精華
0
威望
0
贡献
0
在線時間
0 小時
註冊時間
2011-10-11
發表於 2026-5-9 16:22:45 | 顯示全部樓層
感谢楼主的详细讲解!命名管道确实是Linux进程间通信中非常实用的一个方式,之前一直对fifo似懂非懂,看完你的帖子清晰多了。

[ b ]补充几点小经验[ /b ]:

[ i ]1. 关于阻塞问题[ /i ]
在实际使用中,如果只读端或只写端先打开管道,可能会出现阻塞情况。可以考虑使用O_NONBLOCK标志来避免阻塞,或者在代码中做好同步处理。

[ i ]2. 权限问题[ /i ]
创建fifo时注意umask的设置,否则可能会出现权限不对的情况。楼主的代码中umask(0)处理得很好。

[ i ]3. 实际应用场景[ /i ]
命名管道常用于父子进程之外的独立进程间通信,比如守护进程和客户端之间的通信,比匿名管道更灵活。

[ code ]// 也可以用shell创建命名管道
mkfifo /tmp/myfifo[ /code ]

另外,如果想要实现双向通信,可能需要创建两个fifo管道,这个在后续可以作为一个进阶话题分享。

总之很实用的一篇教程,收藏了!希望能看到更多关于进程间通信的文章,比如socket、共享内存这些~
回覆

使用道具 舉報

您需要登錄後才可以回帖 登錄 | 立即注册

本版積分規則

相关侵权、举报、投诉及建议等,请发 E-mail:qiongdian@foxmail.com

Powered by Discuz! X5.0 © 2001-2026 Discuz! Team.

在本版发帖返回顶部