百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 优雅编程 > 正文

进程间通信(一)—管道

sinye56 2024-11-24 21:29 1 浏览 0 评论

我会用几篇博客总结一下在Linux中进程之间通信的几种方法,我会把这个开头的摘要部分在这个系列的每篇博客中都打出来

进程之间通信的方式

  • 管道
  • 消息队列
  • 信号
  • 信号量
  • 共享存储区
  • 套接字(socket)

在以一切皆文件为原则的Linux系统中,管道也是一种文件(特殊文件),可以使用mkfifo命令创建一个管道文件

在管道文件的前面有一个p来标识管道文件

这次主要说的是通过管道完成进程之间的通信,通过管道通信有两种方式。

一种是匿名管道,一种是命名管道

先来看一段代码

 1 #define MAXLINE 80
 2 int main
 3 {
 4     int n;
 5     int fd[2];
 6     pid_t pid;
 7     char line[MAXLINE];
 8     if (pipe(fd) < 0)
 9         perror("pipe");
10     if ((pid = fork) < 0)
11         perror("fork");
12     if (pid > 0)
13     {
14         //father
15         close(fd[0]);
16 } 17 else 18 { 19close(fd[1]);
20 n = read(fd[0], line, MAXLINE); 21 write(stdout, line, n);//写到标准输出上看一下效果 22 } 23 return 0; 24 25 }

这个程序就是一个简单的父子进程之间通过管道进行通信的一个例子,具体的工作过程我用画图的方式展现出来

注意这一个步骤是十分重要的,如果不关闭相应的端口,就无法正确操作管道。

匿名管道主要利用了,创建子进程的时候会把父进程的文件描述符表拷贝一份这个特征,通过这个特征,父子进程就看到了一个公共的资源—管道,并同时拥有对该管道腹泻的权利,那么一方读,一方写,就可以完成进程之间的通信了。

所谓的匿名管道就是说,没有名字。。。你根本不知道这个管道文件存放在哪,也不知道这个管道文件的文件名,唯一跟这个跟管道文件有联系的只有父子进程中的文件描述符。那么根据匿名管道的实现机制,很容易就能看出他的优缺点。

  • 管道的n个特征
    • 管道是依赖于文件系统的,创建好管道之后,一定要关闭不使用的读写端
    • 只有父子进程才可以使用管道通信,也就是所谓的有血缘关系的进程进行进程间通信。(匿名管道独有)
    • 管道是基于数据流的,面向字节流!(后面会提到消息队列是面向数据块的,对比来看会好懂一些)
    • 管道只能称之为单向数据通信,连半双工都算不上
    • 同步与互斥问题不需要考虑了,管道已经考虑了
    • 当父子进程退出的时候,管道的生命周期就结束了,也就是说管道的生命周期就是进程

上述就是匿名管道的使用和实现机制,可以看出必须有“亲缘关系”的进程之间才可以使用匿名管道来完成进程间通信。父子进程当然可以,“孙子”进程也是可以的~

那么为了解决只有有亲缘关系的进程才能使用这种方式进行通信的弊端,就有了命名管道的通信方式

简单的来说,我们刚才使用的匿名管道是因为不知道文件名和存放路径,所以只能通过继承文件描述符表来获得跟匿名管道建立联系的方式,如果我们知道路径和管道文件名呢?那不就可以完成非亲缘关系的进程间通信了么

  • 函数原型 int mkfifo(const char *pathname,mode_t mode)
  • 其中第一个参数pathname就是路径了,mode 就是创建的管道的访问权限umask
  • 头文件:#include <sys.stat.h> #include <sys/types.h>
  • 返回值,成功返回0,失败返回-1

下面贴上代码

这个是server的

    1 #include <sys/types.h> 
    2 #include <sys/stat.h> 
    3 #include <stdio.h> 
    4 #include <unistd.h> 
    5 #include <stdlib.h> 
    6 #include <errno.h> 
    7 #include <string.h> 
    8 #include <fcntl.h> 
    9 #define _PATH_NAME_ "./pp" 
   10 
   11 int main 
   12 { 
>> 13   char *str="hello world"; 
   14   int fd; 
   15   if(mkfifo(_PATH_NAME_,0644)<0)· 
   16   { 
   17     printf("hahaha\n"); 
   18     printf("mkfifo error %d:%s\n",errno,strerror(errno)); 
   19   } 
   20   fd=open(_PATH_NAME_,O_WRONLY); 
   21   write(fd,str,strlen(str)); 
   22   return 0; 
   23 } 

这个是client的

  2 #include <sys/types.h>
  3 #include <sys/stat.h>
  4 #include <stdio.h>
  5 #include <unistd.h>
  6 #include <stdlib.h>
  7 #include <errno.h>
  8 #include <string.h>
  9 #include <fcntl.h>
 10    
 11 #define _PATH_NAME_ "./pp"
 12    
 13    
 14 int main
 15 {  
 16   int fd;
 17   fd=open(_PATH_NAME_,O_RDONLY); 
 18   char buf[1024]={'\0'};
 19   read(fd,buf,1024);
 20   printf("%s\n",buf);
 21   return 0;
 22 }  

可以通过open函数来打开分别使用read和write函数来读写管道,当然也可以close掉标准输入输出,做成重定向的那样也是可以的

那么管道有多大呢,我们可以往里面扔多少数据呢?

既然管道也是一个文件,那么肯定有大小上限,这也是其一个缺点,大小有限的,那么我们究竟可以写多少数据放进管道呢,使用man 7 pipe命令打开pipe的说明文档,可以看见这么一段

Pipe capacity

A pipe has a limited capacity. If the pipe is full, then a write(2)

will block or fail, depending on whether the O_NONBLOCK flag is set

(see below). Different implementations have different limits for the

pipe capacity. Applications should not rely on a particular capacity:

an application should be designed so that a reading process consumes

data as soon as it is available, so that a writing process does not

remain blocked.

In Linux versions before 2.6.11, the capacity of a pipe was the same as

the system page size (e.g., 4096 bytes on i386). Since Linux 2.6.11,

the pipe capacity is 65536 bytes.

看,我现在的版本是2.6.11之后了,所以是65536字节,之间的版本就是4096字节

最后的最后,来一个管道最大的缺点,管道是基于文件系统的!所以不管是读还是写,都要求访问磁盘进行I/O操作,I/O的速度你懂得,特别慢,所以不适合做多个client的结构,不然会很慢

相关推荐

RHEL8和CentOS8怎么重启网络

本文主要讲解如何重启RHEL8或者CentOS8网络以及如何解决RHEL8和CentOS8系统的网络管理服务报错,当我们安装好RHEL8或者CentOS8,重启启动网络时,会出现以下报错:...

Linux 内、外网双网卡路由配置

1.路由信息的影响Linux系统中如果有多张网卡的情况下,如果路由信息配置不正确,...

Linux——centos7修改网卡名

修改网卡名这个操作可能平时用不太上,可作为了解。修改网卡默认名从ens33改成eth01.首先修改网卡配置文件名(建议将原配置文件进行备份)...

CentOS7下修改网卡名称为ethX的操作方法

?Linux操作系统的网卡设备的传统命名方式是eth0、eth1、eth2等,而CentOS7提供了不同的命名规则,默认是基于固件、拓扑、位置信息来分配。这样做的优点是命名全自动的、可预知的...

Linux 网卡名称enss33修改为eth0

一、CentOS修改/etc/sysconfig/grub文件(修改前先备份)为GRUB_CMDLINE_LINUX变量增加2个参数(net.ifnames=0biosdevname=0),修改完成...

CentOS下双网卡绑定,实现带宽飞速

方式一1.新建/etc/sysconfig/network-scripts/ifcfg-bond0文件DEVICE=bond0IPADDR=191.3.60.1NETMASK=255.255.2...

linux 双网卡双网段设置路由转发

背景网络情况linux双网卡:网卡A(ens3)和网卡B(...

Linux-VMware设置网卡保持激活

Linux系统只有在激活网卡的状态下才能去连接网络,进行网络通讯。修改配置文件(永久激活网卡)...

VMware虚拟机三种网络模式

01.VMware虚拟机三种网络模式由于linux目前很热门,越来越多的人在学习linux,但是买一台服务放家里来学习,实在是很浪费。那么如何解决这个问题?虚拟机软件是很好的选择,常用的虚拟机软件有v...

Rocky Linux 9/CentOS Stream 9修改网卡配置/自动修改主机名(实操)

推荐...

2023年最新版 linux克隆虚拟机 解决网卡uuid重复问题

问题描述1、克隆了虚拟机,两台虚拟机里面的ip以及网卡的uuid都是一样的2、ip好改,但是uuid如何改呢?解决问题1、每台主机应该保证网卡的UUID是唯一的,避免后面网络通信有问题...

Linux网卡的Vlan配置,你可能不了解的玩法

如果服务器上连的交换机端口已经预先设置了TRUNK,并允许特定的VLAN可以通过,那么服务器的网卡在配置时就必须指定所属的VLAN,否则就不通了,这种情形在虚拟化部署时较常见。例如在一个办公环境中,办...

Centos7 网卡绑定

1、切换到指定目录#备份网卡数据cd/etc/sysconfig/network-scriptscpifcfg-enp5s0f0ifcfg-enp5s0f0.bak...

Linux搭建nginx+keepalived 高可用(主备+双主模式)

一:keepalived简介反向代理及负载均衡参考:...

Linux下Route 路由指令使用详解

linuxroute命令用于显示和操作IP路由表。要实现两个不同子网之间的通信,需要一台连接两个网络的路由器,或者同时位于两个网络的网关来实现。在Linux系统中,设置路由通常是为了解决以下问题:该...

取消回复欢迎 发表评论: