Linux命名管道进程通信

简介: Linux命名管道进程通信

前言

命名管道 是实现进程间通信的强大工具,它提供了一种简单而有效的方式,允许不同进程之间进行可靠的数据交换。不仅可以在同一主机上的不相关进程间进行通信,还可以在不同主机之间的进程进行网络通信。


一、什么是命名管道通信

命名管道 是一种半双工的通信机制,用于同一主机上的不相关进程之间的通信。它遵循 FIFO的原则,也就是先写入的数据会先被读取。命名管道在文件系统中有一个唯一的路径名,使用时需要先创建管道,然后不同进程通过打开同一个路径名的管道文件来进行通信。

上篇文章中使用 pipe 函数进行通信的也叫做匿名管道。Linux 进程间通信, 管道

pipe 创建的管道 只能用于 “ 有血缘关系 ” 的进程之间。 但是 FIFO 可以允许 不相关的进程间交换数据。

二、创建方式

  1. 命令行创建: mkfifo 管道名。

  2. 库函数: int mkfifo(const char *pathname, mode_t mode);
    返回值:成功返回 0; 失败 返回 -1。

一旦使用 mkfifo 创建一个 FIFO ,就可以使用 open 打开它,常见的 I/O 函数都可用于 fifo 。如: close , read , write , unlink 等。

三、代码示例

下列代码用来 实现 2 个不相关程的通信,一个用来 写数据,一个用来读数据。

首先使用 命令行创建 管道。

写数据进程:

void sys_error(const char *str)
{
  perror(str);
  exit(1);                // 正常退出程序
}
int main(int argc, char **argv)
{
  int fd, i=0, ret;
  char buff[20];
  if(argc <2)             // 输入两个参数:可执行文件,刚刚创建的 fifo 名字
  {
    printf("Usage: ./a.out fifoname\n");
    return -1;
  }
  fd = open(argv[1], O_WRONLY);       // 以只写的方式打开文件
  if(fd < 0)
  {
    sys_error("open error");
  }
  while(1)
  {
    sprintf(buff, "hello : %d\n", i++);
    write(fd, buff, strlen(buff));          // 不断 写入数据
    sleep(1);                   //休眠 1 秒
  }
  close(fd);
  return 0;
}

读数据进程:

void sys_error(const char *str)
{
  perror(str);
  exit(1);                      // 正常终止程序
}
int main(int argc, char **argv)
{
  int fd, i=0, ret;
  char buff[20];
  if(argc <2)
  {
    printf("Usage: ./a.out fifoname\n");        
    return -1;
  }
  fd = open(argv[1], O_RDONLY);           // 以只读的方式打开文件
  if(fd < 0)
  {
    sys_error("open error");
  }
  while(1)
  {
    ret = read(fd, buff, sizeof(buff));       // 读出数据放入 buff
    write(STDOUT_FILENO, buff, ret);        // 见数据写入 标准输出
  }
  close(fd);
  return 0;
}

注意

对于匿名管道(Anonymous Pipe),它只能有一个读端和一个写端。这意味着一般情况下,只能有一个进程从管道中读取数据,而另一个进程则负责向管道中写入数据。

而对于有名管道(Named Pipe),它可以以支持多个读端和多个写端。这使得多个进程能够同时从管道中读取数据或向管道中写入数据。这样可以实现进程间的并发通信。

例如:

可以将上面示例代码加一个读端:

可以看到 1处 读出的数据,在 2处 就读不出了。因为,管道中数据不可反复读取。一旦读走,管道中不在存在

四、文件进程通信

使用 文件也可以完成 IPC ,理论依据是, fork 后,父子线程共享文件描述符。也就打开了共享文件。


总结

进程间通信是操作系统和分布式系统领域的重要内容,深入了解和掌握不同的进程间通信方式对于开发高效可靠的应用程序至关重要。

相关文章
|
17天前
|
存储 Unix Linux
进程间通信方式-----管道通信
【10月更文挑战第29天】管道通信是一种重要的进程间通信机制,它为进程间的数据传输和同步提供了一种简单有效的方法。通过合理地使用管道通信,可以实现不同进程之间的协作,提高系统的整体性能和效率。
|
17天前
|
消息中间件 存储 供应链
进程间通信方式-----消息队列通信
【10月更文挑战第29天】消息队列通信是一种强大而灵活的进程间通信机制,它通过异步通信、解耦和缓冲等特性,为分布式系统和多进程应用提供了高效的通信方式。在实际应用中,需要根据具体的需求和场景,合理地选择和使用消息队列,以充分发挥其优势,同时注意其可能带来的复杂性和性能开销等问题。
|
1月前
|
网络协议 Linux 网络性能优化
Linux C/C++之TCP / UDP通信
这篇文章详细介绍了Linux下C/C++语言实现TCP和UDP通信的方法,包括网络基础、通信模型、编程示例以及TCP和UDP的优缺点比较。
37 0
Linux C/C++之TCP / UDP通信
|
1月前
|
网络协议 Linux
linux学习之套接字通信
Linux中的套接字通信是网络编程的核心,允许多个进程通过网络交换数据。套接字提供跨网络通信能力,涵盖本地进程间通信及远程通信。主要基于TCP和UDP两种模型:TCP面向连接且可靠,适用于文件传输等高可靠性需求;UDP无连接且速度快,适合实时音视频通信等低延迟场景。通过创建、绑定、监听及读写操作,可以在Linux环境下轻松实现这两种通信模型。
37 1
|
1月前
|
存储 Python
Python中的多进程通信实践指南
Python中的多进程通信实践指南
23 0
|
2月前
|
Java Android开发 数据安全/隐私保护
Android中多进程通信有几种方式?需要注意哪些问题?
本文介绍了Android中的多进程通信(IPC),探讨了IPC的重要性及其实现方式,如Intent、Binder、AIDL等,并通过一个使用Binder机制的示例详细说明了其实现过程。
326 4
|
2月前
|
消息中间件 Unix Linux
C语言 多进程编程(二)管道
本文详细介绍了Linux下的进程间通信(IPC),重点讨论了管道通信机制。首先,文章概述了进程间通信的基本概念及重要性,并列举了几种常见的IPC方式。接着深入探讨了管道通信,包括无名管道(匿名管道)和有名管道(命名管道)。无名管道主要用于父子进程间的单向通信,有名管道则可用于任意进程间的通信。文中提供了丰富的示例代码,展示了如何使用`pipe()`和`mkfifo()`函数创建管道,并通过实例演示了如何利用管道进行进程间的消息传递。此外,还分析了管道的特点、优缺点以及如何通过`errno`判断管道是否存在,帮助读者更好地理解和应用管道通信技术。
|
2月前
|
SQL 网络协议 数据库连接
已解决:连接SqlServer出现 provider: Shared Memory Provider, error: 0 - 管道的另一端上无任何进程【C#连接SqlServer踩坑记录】
本文介绍了解决连接SqlServer时出现“provider: Shared Memory Provider, error: 0 - 管道的另一端上无任何进程”错误的步骤,包括更改服务器验证模式、修改sa用户设置、启用TCP/IP协议,以及检查数据库连接语句中的实例名是否正确。此外,还解释了实例名mssqlserver和sqlserver之间的区别,包括它们在默认设置、功能和用途上的差异。
|
4月前
|
运维 关系型数据库 MySQL
掌握taskset:优化你的Linux进程,提升系统性能
在多核处理器成为现代计算标准的今天,运维人员和性能调优人员面临着如何有效利用这些处理能力的挑战。优化进程运行的位置不仅可以提高性能,还能更好地管理和分配系统资源。 其中,taskset命令是一个强大的工具,它允许管理员将进程绑定到特定的CPU核心,减少上下文切换的开销,从而提升整体效率。
掌握taskset:优化你的Linux进程,提升系统性能
|
4月前
|
弹性计算 Linux 区块链
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
170 4
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
下一篇
无影云桌面