管道是无名的,只能用于有血缘关系之间的进程通讯。FIFO正是为了弥补这个缺陷而设计出来的。服务器
#include <unistd.h>设计
int pipe(int fd[2]);code
经过宏S_ISFIFO来肯定一个描述符或文件是不是FIFO,惟一参数是stat结构体的st_mode成员。server
下面一个例子:进程
父进程使用客户端程序,经过标准输入调用一个路径名来,而后经过管道发送给子进程,子进程使用服务器程序,经过管道获取这个路径名,而后将这个文件内容经过管道再发送给客户端。客户端再将文件内容打印出来。ip
代码:get
/*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); } }