管道是两个进程的连接器,单向,一端作为输入。普通管道只能用于父子进程的进程间通信,命名管道可以实现不同进程间的通信。
创建管道函数
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<wait.h>
#include<errno.h>
#include<signal.h>
#include <setjmp.h>
#include<sys/types.h>
#include<fcntl.h>
#include<sys/stat.h>
/*int pipe(int pipefd[2]);
参数保存系统返回的描述符号,0读,1写
返回0表示成功
返回-1表示函数执行失败
*/
/*首先创建了一个管道,然后创建了子进程,利用该管道从父进程向子进程发送信息。消息是程序的命令行参数,子进程收到父进程传送的消息后输出。*/
/*本例里面每个文件描述符都被关闭了两次,是因为在多进程的环境下,一个有效的管道描述符在并发子进程时会将被所有进程共享的文件表里面的引用次数+1,这里的每一次
都是将其引用计数-1在一个描述符的引用次数为0情况下才会真正的关闭一个描述符*/
int main(int argc, char *argv[])
{
int pfd[2]; //保存管道的文件描述符
pid_t cpid;//保存进程标示符
char buf;//定义变量
if(argc != 2)
{
fprintf(stderr,"Usage: %s <string>\n",argv[0]);//提示用法信息
exit(0);
}
if (pipe(pfd) == -1)
{
perror("pipe");
exit(EXIT_FAILURE);
}
cpid = fork();//创建进程
if (cpid == -1)
{
perror("fork");
exit(EXIT_FAILURE);
}
if (cpid == 0)//子进程
{
close(pfd[1]);//关闭写管道描述符
while (read(pfd[0], &buf, 1) > 0)//管道循环读数据
write(STDOUT_FILENO, &buf, 1);//输出读到数据
write(STDOUT_FILENO, "\n", 1);//输出从管道里读取的数据
close(pfd[0]);//退出子进程
exit(EXIT_SUCCESS);
}
else
{
close(pfd[0]);//关闭文件描述符
write(pfd[1], argv[1], strlen(argv[1]));//向管道写入命令行参数1
close(pfd[1]);//关闭写管道描述符
wait(NULL);//等待子进程退出
exit(EXIT_SUCCESS);
}
return 0;
}