博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
进程子进程linux系统编程之管道(一):匿名管道和pipe函数
阅读量:5084 次
发布时间:2019-06-13

本文共 3539 字,大约阅读时间需要 11 分钟。

最近一直在研究进程子进程之类的问题,现在正好有机会和大家共享一下.

    一、进程间通信

    每个进程各自有不同的用户地址空间,任何一个进程的全局变量在另一个进程中都看不到,所以进程之间要交换数据必须通过内核,在内核中开拓一块缓冲区,进程1把数据从用户空间拷到内核缓冲区,进程2再从内核缓冲区把数据读走,内核供给的这类机制称为进程间通信(IPC,InterProcess Communication)。如下图所示。

    进程和子进程

    二、管道是一种最基本的IPC机制,由pipe函数创建:

#include <unistd.h>

    int pipe(int filedes[2]);

    

调用pipe函数时在内核中开拓一块缓冲区(称为管道)用于通信,它有一个读端一个写端,然后通过filedes参数传出给用户程序两个文件描述符,filedes[0]指向管道的读端,filedes[1]指向管道的写端(很好记,就像0是标准输入1是标准输出一样)。所以管道在用户程序看起来就像一个打开的文件,通过read(filedes[0]);或者write(filedes[1]);向这个文件读写数据实际上是在读写内核缓冲区。pipe函数调用胜利返回0,调用失败返回-1。

    开拓了管道之后如何实现两个进程间的通信呢?比如可以按下面的步骤通信。

    进程和子进程

    

    示例程序如下:

    

    

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
 
/*************************************************************************
    > File Name: process_.c
    > Author: Simba
    > Mail: dameng34@163.com
    > Created Time: Sat 23 Feb 2013 02:34:02 PM CST
 ************************************************************************/
#include<sys/types.h>
#include<sys/stat.h>
#include<unistd.h>
#include<fcntl.h>
#include<stdio.h>
#include<stdlib.h>
#include<errno.h>
#include<string.h>
#include<signal.h>
#define ERR_EXIT(m) \
    
do { \
        perror(m); \
        exit(EXIT_FAILURE); \
    } 
while(
0)
int main(
int argc, 
char *argv[])
{
    
int pipefd[
2];
    
if (pipe(pipefd) == -
1)
        ERR_EXIT(
"pipe error");
    pid_t pid;
    pid = fork();
    
if (pid == -
1)
        ERR_EXIT(
"fork error");
    
if (pid == 
0)
    {
        close(pipefd[
0]);
        write(pipefd[
1], 
"hello"
5);
        close(pipefd[
1]);
        exit(EXIT_SUCCESS);
    }
    close(pipefd[
1]);
    
char buf[
10] = {
0};
    read(pipefd[
0], buf, 
10);
    printf(
"buf=%s\n", buf);
    
return 
0;
}

    1. 父进程调用pipe开拓管道,得到两个文件描述符指向管道的两头。

    2. 父进程调用fork创建子进程,那么子进程也有两个文件描述符指向统一管道。

    3. 父进程关闭管道写端,子进程关闭管道读端。子进程可以往管道里写,父进程可以从管道里读,管道是用环形队列实现的,数据从写端流入从读端流出,这样就实现了进程间通信。

    

    每日一道理
胜利的花朵开放在啊勤劳的枝头,失败的苦果孕育在懒惰的温床之中。

    三、利用pipe和dup2函数模拟命令行 ls  | wc -w 功能

    

    

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
 
/*************************************************************************
    > File Name: process_.c
    > Author: Simba
    > Mail: dameng34@163.com
    > Created Time: Sat 23 Feb 2013 02:34:02 PM CST
 ************************************************************************/
#include<sys/types.h>
#include<sys/stat.h>
#include<unistd.h>
#include<fcntl.h>
#include<stdio.h>
#include<stdlib.h>
#include<errno.h>
#include<string.h>
#include<signal.h>
#define ERR_EXIT(m) \
    
do { \
        perror(m); \
        exit(EXIT_FAILURE); \
    } 
while(
0)
int main(
int argc, 
char *argv[])
{
    
int pipefd[
2];
    
if (pipe(pipefd) == -
1)
        ERR_EXIT(
"pipe error");
    pid_t pid;
    pid = fork();
    
if (pid == -
1)
        ERR_EXIT(
"fork error");
    
if (pid == 
0)
    {
        dup2(pipefd[
1], STDOUT_FILENO); 
//输出重定向
        close(pipefd[
1]);
        close(pipefd[
0]);
        execlp(
"ls"
"ls"
NULL);
        fprintf(stderr, 
"error execute ls\n");
        exit(EXIT_FAILURE);
    }
    dup2(pipefd[
0], STDIN_FILENO);
    close(pipefd[
0]);
    close(pipefd[
1]);
    execlp(
"wc"
"wc"
"-w"
NULL);
    fprintf(stderr, 
"error execute wc\n");
    exit(EXIT_FAILURE);
    
return 
0;
}

    我们知道命令行 ls | wc -w 中 ls 会输出到管道,而wc 从管道里读取,现在使用dup2复制文件描述符,使ls 的标准输出为管道,wc 的标准输入也为管道,即使父进程先被调度,因为默认是阻塞I/O操纵,故wc 会read 阻塞直到管道被子进程写入了数据。

    

    使用管道有一些限制:

    两个进程通过一个管道只能实现单向通信,比如最下面的例子,父进程读子进程写,如果有时候也须要子进程读父进程写,就必须另开一个管道。

    管道的读写端通过打开的文件描述符来传递,因此要通信的两个进程必须从它们的公共先人那里继承管道文件描述符。下面的例子是父进程把文件描述符传给子进程之后父子进程之间通信,也可以父进程fork两次,把文件描述符传给两个子进程,然后两个子进程之间通信,总之须要通过fork传递文件描述符使两个进程都能拜访统一管道,它们才能通信。

文章结束给大家分享下程序员的一些笑话语录: 问:你觉得让你女朋友(或者任何一个女的)从你和李彦宏之间选一个,你觉得她会选谁?  

  答:因为李艳红这种败类,所以我没女友!

转载于:https://www.cnblogs.com/xinyuyuanm/archive/2013/05/20/3089395.html

你可能感兴趣的文章
nginx常见内部参数,错误总结
查看>>
对象与类
查看>>
《奸的好人2》财色战场----笔记
查看>>
BZOJ 1834网络扩容题解
查看>>
bzoj1878
查看>>
【Vegas原创】Mysql绿色版安装方法
查看>>
Thrift Expected protocol id ffffff82 but got 0
查看>>
分享《去哪儿网》前端笔试题
查看>>
2013-07-04学习笔记二
查看>>
CP15 协处理器寄存器解读
查看>>
【codeforces 787B】Not Afraid
查看>>
【9111】高精度除法(高精度除高精度)
查看>>
【hihocoder 1312】搜索三·启发式搜索(普通广搜做法)
查看>>
JavaFX中ObservableValue类型
查看>>
杭电 1097 A hard puzzle
查看>>
[转载]INFORMIX锁机制及如何剖析其锁申辩(第二部门)
查看>>
Andriod-项目stymqjlb-学习笔记2-原型
查看>>
Web AppDomain
查看>>
JQuery创建规范插件
查看>>
AD 域服务简介(三)- Java 对 AD 域用户的增删改查操作
查看>>