一次课就把进程间通信讲完了,不得不说是非常的快,说到头来还是归于本学期的课程安排太赶了。这里复习一下我上一次学习的所有进程间通信的知识点。
管道
命名管道(FIFO)
上一节课讲过管道部分中的匿名管道,而匿名管道只能父子进程(有亲缘关系的进程)通信,并且匿名管道只能单向传递,即一个进程写,一个进程读,如果要同时进行就得重新再开一个管道。所以我们引入一个能解决这些局限的另一个管道----命名管道。
与管道的区别:提供了一个路径名与之关联,以FIFO文件的形式存储于文件系统中,能够实现任何两个进程之间通信。而匿名管道对于文件系统是不可见的,它仅限于在父子进程之间的通信。
FIFO是一个设备文件,在文件系统中以文件名的形式存在,因此即使进程与创建FIFO的进程不存在血缘关系也依然可以通信,前提是可以访问该路径。
FIFO(first input first output)总是遵循先进先出的原则,即第一个进来的数据会第一个被读走。
创建格式
#include <sys/stat.h>
int mknod(const char* path, mode_t mod, dev_t dev);
int mkfifo(const char* path, mode_t mod);
课程只讲了mkfifo,所以mknod就不说了。
mkfifo函数的作用:
创建命名管道(FIFO文件),命名管道与系统中的一个路径名关联,以文件的形式存在于文件系统中,通过FIFO的路径名访问FIFO文件,实现进程间通信。成功返回0,失败返回-1。
参数:
pathname:管道文件的路径名,通过FIFO路径名访问FIFO文件;
mode:指定FIFO的权限;
案例
使用FIFO实现没有亲缘关系进程间的通信。没有亲缘关系的进程间通信,需要两段程序来实现:
- fifo_write.c实现FIFO的写操作;
- fifo_read.c实现FIFO的读操作。
//fifo_write.c
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
int main(int paraArgc,char *paraArgv[]){
if(paraArgc < 2){ //判断是否传入文件名
printf("./a.out fifoname\n");
exit(1);
}//of if
int tempRet = access(paraArgv[1], F_OK); //判断fifo文件是否存在
if(tempRet == -1){ //若fifo不存在就创建fifo
int tempFIFO = mkfifo(paraArgv[1], 0664);