进程间通信
- 1.进程间通信介绍
- 2.匿名命名管道原理操作
1.进程间通信介绍
1.1 进程间通信目的:一个进程需要将他的数据发送给另一个进程,大家应该都多少接触过linux中的管道符"|",这个符号就是用来多个命令执行,在Linux中每一个命令都是一个独立的进程,这里我们可以在Linux中看到这些命令,都是一个个的可执行程序,每次输入时相当于一个可执行程序开始跑。
我们这里切换到ls这个命令的目录下,在ubuntu系统下我们看到了熟悉的apt安装命令,还有很多比较冷门的命令,我们学习阶段接触的比较少,但是我们可以发现这都是一些可执行程序。
这里我们还可以使用ldd命令查一下,这些命令都是用什么写的:
使用ldd命令可以查看这个可执行程序使用的依赖哪些库产生,这里明显的看出是依赖的c语言的动态库编写的。
所以这里我们需要理解,每一个命令都是一个可执行程序。
2.匿名命名管道原理操作
讲到原理,那么为什么可以使用管道符"|"一起执行两个可执行程序呢,这里就是通信的力量啦。
那么什么是通信呢?
这里两个进程就可以指向同一块空间,这样我们就具备了最基本的沟通能力。
有了沟通的能力我们就可以想办法让他们可以传递信息。
这里我们要介绍一个接口pipe。
//此处以C++举例
#include<unistd.h>
int pipe(int fd[2]);
pipe函数定义中的fd参数是一个大小为2的一个数组类型的指针。该函数成功时返回0,并将一对打开的文件描述符值填入fd参数指向的数组。失败时返回 -1并设置errno。
下面是一段验证原理的代码,使用c++编写,最后实现了管道的单向通信:
#include <iostream>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <cstdio>
#include <cstring>
using namespace std;void write_pipe(int wfd)
{char buffer[1024];int cnt = 0;while (true){snprintf(buffer, sizeof(buffer), "I am god! 我的pid:%d, cnt=%d", getpid(), cnt++);ssize_t ret = write(wfd, buffer, strlen(buffer));if (ret < 0){perror("write");break;}sleep(1);}
}void read_pipe(int rfd)
{char buffer[1024];while (true){ssize_t n = read(rfd, buffer, sizeof(buffer) - 1);if (n > 0){buffer[n] = '\0';cout << "child say: " << buffer << endl;}else if (n == 0){cout << "pipe closed" << endl;break;}else{perror("read");break;}}
}int main()
{int fds[2] = {0};if (pipe(fds) < 0){perror("pipe");return 1;}pid_t id = fork();if (id < 0){perror("fork");return 1;}if (id == 0){ // 子进程close(fds[0]); // 关闭读端write_pipe(fds[1]);close(fds[1]);exit(0);}else{ // 父进程close(fds[1]); // 关闭写端read_pipe(fds[0]);close(fds[0]);waitpid(id, nullptr, 0);}return 0;
}
这里我们写了一个shell脚本用来观察我们写的程序,发现是可以进行读写。