进程间通信:是指多个进程之间数据相互转换。
进程间通讯的方式:
单机通讯:信号、管道(有名管道、无名管道)、信号量、消息队列、共享存储
多机通讯:套接字(socket)
管道通讯的原理
有名管道:应用于任意两个进程之间数据的单向传递
管道操作:创建 命令方式:mkfifo 函数方式:mkfifo();
打开 open
读数据 read
写数据 write
关闭 close
因为管道文件创建之后,之后的操作就和普通文件操作一样,那我在此就不再进行具体的说明了。
在此说明一下,有名管道只是在文件目录树中有一个文件标示,它实际上并不占据磁盘空间,而是将数据缓存在内存上。
有名管道特性:阻塞运行
阻塞运行函数:函数调用以后并不会立即返回,需要等待某些条件的发生才会返回。
open操作管道文件
(1)如果一个进程是以只写的方式打开一个管道文件,那么open函数会阻塞运行,直到另一个进程以读的方式(不管是只读还是读写)打开管道文件。open函数才会返回,进程才会接着运行。
(2)如果一个进程是以只读的方式打开一个管道文件,那么open函数会阻塞运行,直到另一个进程以写的方式(不管是只写还是读写)打开管道文件。open函数才会返回,进程才会接着运行。
read函数操作管道文件----read函数操作管道文件时也会阻塞运行,直到写端写入数据或者所有的写端全都关闭。read函数读取数据并且会将内存上的已读数据清空。
下面请看两个代码
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<string.h>
#include<unistd.h>
#include<fcntl.h>
int main()
{
int fd=open("FIFO",O_WRONLY);
assert(fd!=-1);
if(fd==-1)
{
exit(0);
}
printf("FIFO open success\n");
write(fd,"hello world",10);
printf("write success\n");
}
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<string.h>
#include<unistd.h>
#include<fcntl.h>
int main()
{
int fd=open("FIFO",O_RDONLY);
assert(fd!=-1);
if(fd==-1)
{
exit(0);
}
printf("FIFO open success\n");
char buff[128]={0};
read(fd,buff,127);
printf("read data:%s\n",buff);
}
如上面两个代码所示,如果单独执行其中一个代码,那么将会没有结果。要想代码2中顺利输出代码1中输入到管道文件中的数据,那么在运行代码1的同时运行代码2.
测试文件如下:
单独执行代码1:,进程a会阻塞运行,等待另一个进程b以读方式打开
补充执行代码2:后的结果
进程a
进程b
上面的结果显示进程a成功写入数据,进程b也成功读取了数据。两个进程成功通过管道进行了通讯。