管道是无名的,只能用于有血缘关系之间的进程通信。FIFO正是为了弥补这个缺陷而设计出来的。
#include <unistd.h>
int pipe(int fd[2]);
通过宏S_ISFIFO来确定一个描述符或文件是否是FIFO,唯一参数是stat结构体的st_mode成员。
下面一个例子:
父进程使用客户端程序,通过标准输入调用一个路径名来,然后通过管道发送给子进程,子进程使用服务器程序,通过管道获取这个路径名,然后将这个文件内容通过管道再发送给客户端。客户端再将文件内容打印出来。
代码:
/*mainpipe.c*/
int main(int argc, char **argv)
{
int pipe1[2],pipe2[2];
pid_t childpid;
pipe(pipe1);//建立两个管道,管道1用于客户端将路径名发送给服务器端
pipe(pipe2);//管道2用于将服务器端输出内容发送给客户端
if ((childpid = fork()) == 0) {//子进程
close(pipe1[1]);//关闭写,只读管道1
close(pipe2[0]);//关闭读,只写管道2
server(pipe1[0], pipe2[1]);//从第一个参数读,第二个参数写
exit(0);
}
/*parent*/
close(pipe1[0]);
close(pipe2[1]);
client(pipe2[0], pipe1[1]);//从第一个参数读,第二个参数写
waitpid(childpid, NULL, 0);
exit(0);
}
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#define MAXLINE 1000
/*从标准输入读取一个路径名,将这个路径名写入一个管道里去
,再另外一个管道读取服务器端发送过来的内容,然后显示在
标准输出里。第一个参数是读取通道,第二个参数是写入通道*/
//client.c
void client(int readfd, int writefd)
{
size_t len;
ssize_t n;
char buff[MAXLINE];
fgets(buff, MAXLINE, stdin);//从标准输入获取一行,以换行符结尾的
len = strlen(buff);
if (buff[len-1] == '\n')
len--;//去除最后一个换行符
write(writefd, buff, len);//写入到第二个参数里去
while( (n = read(readfd, buff, MAXLINE)) > 0)//从第一个参数读取,如果,
//在服务器端这个管道关闭了,这里将会返回0
write(STDOUT_FILENO, buff, n);
}
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <error.h>
#include <fcntl.h>
#include <errno.h>
#define MAXLINE 1000
/*这是个服务器程序,从第一个参数读取路径名,然后打开这个路径名,
将这个路径名文件内容输入到第二个参数写入通道去*/
//server.c
void server(int readfd, int writefd)
{
int fd;
ssize_t n;
char buff[MAXLINE+1];
if ( (n = read(readfd, buff, MAXLINE)) == 0) //从第一个参数读取路径名
perror("end of file while reading pathname");
buff[n] = '\0';给路径名加一个结尾
if ( (fd = open(buff,O_RDONLY)) < 0) {//打开路径名
snprintf(buff + n, sizeof(buff) - n, ":can't open, %s\n", strerror(errno));
n = strlen(buff);
write(writefd, buff, n);
} else {
while ( (n =read(fd, buff, MAXLINE)) > 0)//读取路径名那个文件内容
write(writefd, buff, n);//将内容写到第二个参数去
close(fd);
}
}