3. 进程通信——管道
Unix
中的管道是一种先进先出
(FIFO)
特殊文件,这种文件的大小是预定义的(版本相关),通常被定义为
10
个逻辑块,每个大小为
512
字节。在系统的
<limits.h>
或
<sys/param.h>
中,包含一个常量
PIPE_BUF
用来说明一个管道缓冲区可以容纳的最大字节数。
管道是半双工的,数据只能向一个方向流动;需要双方通信时,需要建立起两个管道。只能用于父子进程或者兄弟进程之间(具有亲缘关系的进程);
管道对于管道两端的进程而言,就是一个文件,但它不是普通的文件,单独构成一种文件系统,并且只存在于内存中。
数据的读出和写入:一个进程向管道中写的内容被管道另一端的进程读出。写入的内容每次都添加在管道缓冲区的末尾,并且每次都是从缓冲区的头部读出数据。
如果管道的写端不存在,则认为已经读到了数据的末尾,读函数返回的读出字节数为
0
。当管道的写端存在时,如果请求的字节数目大于
PIPE_BUF
,则返回管道中现有的数据字节数,如果请求的字节数目不大于
PIPE_BUF
,则返回管道中现有数据字节数(此时,管道中数据量小于请求的数据量);或者返回请求的字节数(此时,管道中数据量不小于请求的数据量)。
1)
示例程序如下(使用系统调用
pipe()
创建无名管道):
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
main(){
int filedes[2];
char buffer[80];
if(pipe(filedes)<0)
perror(“pipe”);
if(fork()>0){
char s[ ] = “hello!/n”;
close(filedes[0]);
write(filedes[1],s,sizeof(s));
close(filedes[1]);
}else{
close(filedes[1]);
read(filedes[0],buffer,80);
printf(“%s”,buffer);
close(filedes[0]);
}
}
2)
创建有名管道
a)
Shell
命令:
mknod PIPENAME p
b)
系统调用:
int mknod(const char *path, mode_t mode, dev_t dev);
c)
系统调用:
int mkfifo(const char *path, mode_t mode);
mkfifo()
会依参数
pathname
建立特殊的
FIFO
文件,该文件必须不存在,而参数
mode
为该文件的权限,
mkfifo()
建立的
FIFO
文件其他进程都可以用读写一般文件的方式存取。下面是
mkfifo()
的例子:
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
main(){
char buffer[80];
int fd;
unlink(FIFONAME);
mkfifo(FIFONAME,0666);
if(fork()>0){
char s[ ] = “hello!/n”;
fd = open (FIFONAME,O_WRONLY);
write(fd,s,sizeof(s));
close(fd);
}else{
fd= open(FIFONAME,O_RDONLY);
read(fd,buffer,80);
printf(“%s”,buffer);
close(fd);
}
}
3)
下面是一个
server&client
通过管道通信的例子(后篇)