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

本文共 3468 字,大约阅读时间需要 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传递文件描述符使两个进程都能拜访统一管道,它们才能通信。

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

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

你可能感兴趣的文章
pfSense LAGG(链路聚合)设置
查看>>
教学思路SQL之入门习题《学生成绩》 七.存储过程基础知识
查看>>
createrepo 无法使用解决
查看>>
.net安全类库
查看>>
在Windows 2008 R2上部署SCCM 2007 R2
查看>>
tablespace backup模式一个没用的技术
查看>>
PostgreSQL安装
查看>>
七牛实时音视频云视频连线demo(web部分)
查看>>
Mysql 权限
查看>>
Spring事务管理(详解+实例)
查看>>
ubuntu apt-get install 出现无法定位软件包...
查看>>
centos7 下 基于docker搭建java/tomcat (方式一)
查看>>
全世界最好的编辑器VIM之Windows配置(gvim)[未测试]
查看>>
2018年你需要知道的13个JavaScript工具库
查看>>
当你点击按钮的时候如何设置其他按钮不可点击
查看>>
spring 高级装配
查看>>
【合集】parasoft Jtest 从安装到使用教程合集,收藏推荐!
查看>>
Python Pygame库的学习之路(1)
查看>>
信息安全与Linux系统
查看>>
Ubuntu安装mysql
查看>>