常用的进程通信方式有:
传统的进程通信方式:无名管道(pipe),有名管道(fifo),信号(signal)
system V IPC对象:共享内存(share memeory),消息队列(message queue),信号灯(semaphore)
BSD:套接字(socket)
pipe:pipe只能用于具有亲緣关系的进程之间通信
半双工通信模式,具有固定的读端和写端
管道可以看作是一种特殊的文件,对于它的读写可以使用文件IO如read,write函数。
当管道无数据时,读操作阻塞
写数据时,linux不保证写入原子性,管道缓冲区一有空闲区域,写进程就会试图向管道缓冲区写数据,如果读操作不读取缓冲区数据,那么写操作将会一直阻塞。
读端存在,写数据才有意义,否则管道中写入数据会收到内核传来的SIFPIPE信号
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<errno.h>
int main()
{
int n,fd[2];
pid_t pid;
char line[100];
if(pipe(fd)<0)//pipe产生两个文件描述符,fd[0]读取,fd[1]写
{
perror("pipe");
exit(1);
}
if((pid=fork())<0)//fork产生父子进程
{
perror("fork");
exit(1);
}
if(pid>0)//父进程写,关闭读
{
close(fd[0]);
write(fd[1],"hello world\n",12);
}
else//子进程读,关闭写
{
close(fd[1]);
n=read(fd[0],line,100);
write(STDOUT_FILENO,line,n);//打印屏幕
}
exit(0);
}
fifo:
无名管道可以使两个不相干的进程通信,使用路径来指出,在文件系统中可见
使用I/O操作
遵循先进先出规则
不支持Iseek()操作
! 管道一个以读方式打开,一个以写方式打开,否则,进入阻塞
1:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<fcntl.h>
#include<sys/types.h>
#include<sys/stat.h>
int main(int argc,char **argv)
{
int fd;
char buf[20]="hello world!\n";
if((mkfifo("my_fifo",O_CREAT|O_RDWR|0666))<0)//创建有名管道
{
perror("mkfifo");
exit(1);
}
if((fd=open("my_