Linux进程 线程 进程间通信 IPC——管道
进程间通信(IPC机制)的方法:
(1)管道 (2)信号量 (3)共享内存 (4)消息队列 (5)套接字
有名管道
管道的分类:有名管道和无名管道 这其中有名管道也叫做命名管道
区别:有名管道在任意两个进程之间通信。而无名管道是在父子进程之间通信。
创建有名管道使用命令:mkfifo
打开管道:open();
关闭管道:close();
读数据:read();
写入数据:write();
一般来说,如果进程a要从键盘获取数据传递给另一个进程b,需要使用一个文件。而这个所带来的问题就是,很慢并且读数据时,对于a什么时候会进行写入时未知的。
管道创建之后,他会在内存上分配一块空间。所以,管道的大小永远为0;其中管道一个是读打开,一个是写打开。
//a.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <assert.h>
#include <string.h>int main()
{int fd=open("fifo",O_WRONLY);assert(fd!=-1);printf("fd=%d\n",fd);write(fd,"hello",5); close(fd);
}//b.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <assert.h>int main()
{int fd=open("./fifo",O_RDONLY);assert(fd!=-1);printf("fd=%d\n",fd); char buff[128]={0};read(fd,buff,127);printf("read:%s\n",buff);close(fd);exit(0);
}
管道的特点:
(1)管道必须读,写进程同时open,否则会阻塞;
(2)如果管道没有数据,那么read会阻塞;
(3)管道的写端关闭,读read返回值为0
(4)管道打开的时候只有只读和只写两种方式,读写方式打开是未定义的。
无名管道
无名管道:父子进程间通信
创建无名管道的命令:pipe
int pipe (int pipefd[2]);
调用pipe系统调用之后,返回两个文件描述符,fd[0]为读端,fd[1]为写端;
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <assert.h>
#include <string.h>int main()
{int fd[2];assert(pipe(fd)!=-1);//fd[0] fd[1] pid_t pid=fork();assert(pid!=-1);if(pid==0){close(fd[1]);char buff[128]={0};read(fd[0],buff,127);printf("child read:%s\n",buff);close(fd[0]);}else{close(fd[0]);write(fd[1],"hello",5);close(fd[1]);}exit(0);
}
管道的特点:
(1)管道必须读,写进程同时open,否则会阻塞;
(2)如果管道没有数据,read会阻塞
(3)管道的写端关闭,读端的返回值为0
(4)管道打开的时候只有只读和只写两种方式,读写方式打开时未定义的
(5)无论有名还是无名,写入管道的数据都在内存中(管道的大小永远为0)
(6)管道是一种半双工通信方式(通信方式有单工,半双工,全双工)
(7)有名管道和无名管道的区别:有名管道可以在任意进程间使用,无名管道主要在父子进程间通信
(8)管道的读端关闭,写会产生异常。