Linux系统编程-进程间通信(管道)

简介: 标准流管道像文件操作有标准io流一样,管道也支持文件流模式。用来创建连接到另一进程的管道popen和pclose。

1. 进程间通信方式介绍

这篇文章介绍Linux下进程的间的通信方式,常用的方式如下:

1. socket—网络通信
2. 管道---无名管道—命名管道---文件--FIFO
3. 消息队列
4. 共享内存
5. 信号量集
6. 信号—signal捕获信号---kill命令发送信号
 int kill(pid_t pid, int sig);

2. 标准流管道

标准流管道像文件操作有标准io流一样,管道也支持文件流模式。用来创建连接到另一进程的管道popen和pclose。
函数原型:

#include <stdio.h>
FILE* popen(const char* command, const char* open_mode);
int pclose(FILE* fp);

popen用于启动进程,用法含义与fopen类似,第二个参数填权限,支持填"r"和"w"。

pclose用于关闭进程,释放资源。

popen启动进程之后可以直接与启动的进程间通信,比较方便。

示例代码: 从标准管道流中读取 打印/etc/profile的内容:

#include <stdio.h>
#include <stdlib.h>
int main()
{
    FILE *file=NULL;
    size_t len=0;
    char buff[1024+1];
    
    file=popen("cat /etc/profile","r"); //执行cat /etc/profile命令,读方式。
  
  if(file!=NULL)
  {
         len=fread(buff,1,1024,file); //读数据    
         buff[len]='\0';
         printf("len=%d\n %s\n",len,buff);
  } 
  pclose(file); //等待线程结束。
  return 0;
}

3. 无名管道

无名管道用于有亲戚关系的进程间通信。 比如: 兄弟进程、父子进程等。

#include <unistd.h> 
int pipe(int fds[2]);

pipe函数用于创建一个无名管道,如果成功,fds[0]就存放可读的文件描述符,fds[1]就存放可写文件描述符。

返回值: 0表示成功,-1表示失败。

无名管道的特点:

  1. 只能在亲缘关系进程间通信(父子或兄弟)
  2. 半双工(固定的读端和固定的写端)
  3. 虚拟管道文件是一个存在内存的特殊文件,可以用read、write函数进行操作。

这里说的管道,就像一条水管,有两个端口,一端进水,另一端出水。管道也有两个端口,分别是读端和写端,进水可看成数据从写端被写入,出水可看数据从读端被读出。在程序里分别就对应了read和write函数。

示例代码:

#include <stdio.h>
#include <unistd.h>
#include <string.h>

int main(int argc,char **argv)
{
    int fds[2];
    /*1. 创建无名管道:得到管道读写文件描述符  fds[0]和读端相对应, fds[1]和写端相对应*/
    pipe(fds);
    /*2. 创建子进程*/
    pid_t pid;
    pid=fork();
    if(pid==0) //子进程
    {
        char buff[100+1];
        int cnt;
        //从管道的读端读取数据
        cnt=read(fds[0],buff,100); //带阻塞功能
        buff[cnt]='\0';
        printf("子进程收到的数据:%d,%s\n",cnt,buff);
    }
    else //父进程
    {
        char buff[]="1234567890";
        //向管道的写端写数据
        write(fds[1],buff,strlen(buff));
        //等待子进程结束
        wait(NULL);
        printf("父进程正常结束.\n");
    }
    return 0;
}

4. 命名管道

无名管道只能在亲缘关系的进程间通信大大限制了管道的使用,有名管道突破了这个限制,通过指定路径名的形式实现不相关进程间的通信,因为命名管道通信使用的管道是一个实体文件,在磁盘上的存在的,而无名管道是存在内存中的虚拟文件,其他进程无法访问,导致没有关联的进程无法进行通信。

4.1 在命令行如何创建管道文件?

[wbyq@wbyq test]$ mkfifo test.fifo
[wbyq@wbyq test]$ ls
test.fifo
[wbyq@wbyq test]$ ls -l
总用量 0
prw-rw-r--. 1 wbyq wbyq 0 10月 15 15:29 test.fifo

4.2 在命令行演示两个相关的进程通过进行管道文件进行通信

image-20211213090725147

4.3 创建fifo文件的函数

1. 创建FIFO文件
#include <sys/types.h>
#include <sys/stat.h>
int mkfifo(const char *pathname, mode_t mode);
参数:
pathname:创建的FIFO文件的全路径名;
mode:文件访问权限,比如0666。
返回值:如果创建成功,则返回0,否则-1。

2. 删除FIFO文件
#include <unistd.h>
int unlink(const char *pathname);

3. 用命令创建和删除FIFO文件
用命令mkfifo创建。
用命令unlink删除。

4.4 创建写端: FIFO

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int main(int argc,char **argv)
{
    if(argc!=2)
    {
        printf("./a.out <fifo文件--写>\n");
        return 0;
    }
    int fd=open(argv[1],O_WRONLY);
    if(fd<0)
    {
        printf("%s 文件打开失败.\n",argv[1]);
        return 0;
    }
    char buff[]="1234567890";
    write(fd,buff,strlen(buff));
    close(fd);
    return 0;
}
示例代码: 进程B负责读数据
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int main(int argc,char **argv)
{
    if(argc!=2)
    {
        printf("./a.out <fifo文件--读>\n");
        return 0;
    }
    int fd=open(argv[1],O_RDONLY);
    if(fd<0)
    {
        printf("%s 文件打开失败.\n",argv[1]);
        return 0;
    }
    char buff[100+1];
    int cnt;
    cnt=read(fd,buff,100);
    buff[cnt]='\0';
    printf("buff=%s\n",buff);
    close(fd);
    return 0;
}

4.5 创建读端: FIFO

在命令行演示两个相关的进程通过进行管道文件进行通信.

image-20211213091027344

代码创建读端,读取数据:

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>

int main(void)
{
    int fd=0;
    int len=0;
    char buff[100];
    fd=open("myfifo",O_RDONLY);
    len=read(fd,buff,100);
    buff[len]='\0';
    printf("read: %s\n",buff);
    return 0;
}
目录
相关文章
|
21天前
|
存储 缓存 监控
Linux缓存管理:如何安全地清理系统缓存
在Linux系统中,内存管理至关重要。本文详细介绍了如何安全地清理系统缓存,特别是通过使用`/proc/sys/vm/drop_caches`接口。内容包括清理缓存的原因、步骤、注意事项和最佳实践,帮助你在必要时优化系统性能。
158 78
|
1天前
|
消息中间件 Linux
Linux:进程间通信(共享内存详细讲解以及小项目使用和相关指令、消息队列、信号量)
通过上述讲解和代码示例,您可以理解和实现Linux系统中的进程间通信机制,包括共享内存、消息队列和信号量。这些机制在实际开发中非常重要,能够提高系统的并发处理能力和数据通信效率。希望本文能为您的学习和开发提供实用的指导和帮助。
35 20
|
25天前
|
Linux Shell 网络安全
Kali Linux系统Metasploit框架利用 HTA 文件进行渗透测试实验
本指南介绍如何利用 HTA 文件和 Metasploit 框架进行渗透测试。通过创建反向 shell、生成 HTA 文件、设置 HTTP 服务器和发送文件,最终实现对目标系统的控制。适用于教育目的,需合法授权。
56 9
Kali Linux系统Metasploit框架利用 HTA 文件进行渗透测试实验
|
21天前
|
存储 监控 Linux
嵌入式Linux系统编程 — 5.3 times、clock函数获取进程时间
在嵌入式Linux系统编程中,`times`和 `clock`函数是获取进程时间的两个重要工具。`times`函数提供了更详细的进程和子进程时间信息,而 `clock`函数则提供了更简单的处理器时间获取方法。根据具体需求选择合适的函数,可以更有效地进行性能分析和资源管理。通过本文的介绍,希望能帮助您更好地理解和使用这两个函数,提高嵌入式系统编程的效率和效果。
89 13
|
28天前
|
SQL 运维 监控
南大通用GBase 8a MPP Cluster Linux端SQL进程监控工具
南大通用GBase 8a MPP Cluster Linux端SQL进程监控工具
|
22天前
|
Ubuntu Linux C++
Win10系统上直接使用linux子系统教程(仅需五步!超简单,快速上手)
本文介绍了如何在Windows 10上安装并使用Linux子系统。首先,通过应用商店安装Windows Terminal和Linux系统(如Ubuntu)。接着,在控制面板中启用“适用于Linux的Windows子系统”并重启电脑。最后,在Windows Terminal中选择安装的Linux系统即可开始使用。文中还提供了注意事项和进一步配置的链接。
40 0
|
8月前
|
Linux Shell 调度
【Linux】7. 进程概念
【Linux】7. 进程概念
75 3
|
8月前
|
存储 缓存 Linux
【Linux】进程概念(冯诺依曼体系结构、操作系统、进程)-- 详解
【Linux】进程概念(冯诺依曼体系结构、操作系统、进程)-- 详解
|
5月前
|
Linux Shell 调度
【在Linux世界中追寻伟大的One Piece】Linux进程概念
【在Linux世界中追寻伟大的One Piece】Linux进程概念
50 1
|
7月前
|
存储 Linux Shell
Linux进程概念(上)
冯·诺依曼体系结构概述,包括存储程序概念,程序控制及五大组件(运算器、控制器、存储器、输入设备、输出设备)。程序和数据混合存储,通过内存执行指令。现代计算机以此为基础,但面临速度瓶颈问题,如缓存层次结构解决内存访问速度问题。操作系统作为核心管理软件,负责资源分配,包括进程、内存、文件和驱动管理。进程是程序执行实例,拥有进程控制块(PCB),如Linux中的task_struct。创建和管理进程涉及系统调用,如fork()用于创建新进程。
71 3
Linux进程概念(上)