Python进程通信之匿名管道实例讲解


Posted in Python onApril 11, 2015

匿名管道

管道是一个单向通道,有点类似共享内存缓存.管道有两端,包括输入端和输出端.对于一个进程的而言,它只能看到管道一端,即要么是输入端要么是输出端.

os.pipe()返回2个文件描述符(r, w),表示可读的和可写的.示例代码如下:

#!/usr/bin/python

import time

import os
def child(wpipe):

    print('hello from child', os.getpid())

    while True:

        msg = 'how are you\n'.encode()

        os.write(wpipe, msg)

        time.sleep(1)
def parent():

    rpipe, wpipe = os.pipe()

    pid = os.fork()

    if pid == 0:

        child(wpipe)

        assert False, 'fork child process error!'

    else:

        os.close(wpipe)

        print('hello from parent', os.getpid(), pid)

        fobj = os.fdopen(rpipe, 'r')

        while True:

            recv = os.read(rpipe, 32)

            print recv
parent()

输出如下:
('hello from parent', 5053, 5054)

('hello from child', 5054)

how are you
how are you
how are you
how are you

我们也可以改进代码,不用os.read()从管道中读取二进制字节,而是从文件对象中读取字符串.这时需要用到os.fdopen()把底层的文件描述符(管道)包装成文件对象,然后再用文件对象中的readline()方法读取.这里请注意文件对象的readline()方法总是读取有换行符'\n'的一行,而且连换行符也读取出来.还有一点要改进的地方是,把父进程和子进程的管道中不用的一端关闭掉.

#!/usr/bin/python

import time

import os
def child(wpipe):

    print('hello from child', os.getpid())

    while True:

        msg = 'how are you\n'.encode()

        os.write(wpipe, msg)

        time.sleep(1)
def parent():

    rpipe, wpipe = os.pipe()

    pid = os.fork()

    if pid == 0:

        os.close(rpipe)

        child(wpipe)

        assert False, 'fork child process error!'

    else:

        os.close(wpipe)

        print('hello from parent', os.getpid(), pid)

        fobj = os.fdopen(rpipe, 'r')

        while True:

            recv = fobj.readline()[:-1]

            print recv
parent()

输出如下:

('hello from parent', 5108, 5109)

('hello from child', 5109)

how are you

how are you

how are you

如果要与子进程进行双向通信,只有一个pipe管道是不够的,需要2个pipe管道才行.以下示例在父进程新建了2个管道,然后再fork子进程.os.dup2()实现输出和输入的重定向.spawn功能类似于subprocess.Popen(),既能发送消息给子进程,由能从子子进程获取返回数据.
#!/usr/bin/python

#coding=utf-8

import os, sys
def spawn(prog, *args):

    stdinFd = sys.stdin.fileno()

    stdoutFd = sys.stdout.fileno()
    parentStdin, childStdout = os.pipe()

    childStdin, parentStdout= os.pipe()
    pid = os.fork()

    if pid:

        os.close(childStdin)

        os.close(childStdout)

        os.dup2(parentStdin, stdinFd)#输入流绑定到管道,将输入重定向到管道一端parentStdin

        os.dup2(parentStdout, stdoutFd)#输出流绑定到管道,发送到子进程childStdin

    else:

        os.close(parentStdin)

        os.close(parentStdout)

        os.dup2(childStdin, stdinFd)#输入流绑定到管道

        os.dup2(childStdout, stdoutFd)

        args = (prog, ) + args

        os.execvp(prog, args)

        assert False, 'execvp failed!'
if __name__ == '__main__':

    mypid = os.getpid()

    spawn('python', 'pipetest.py', 'spam')
    print 'Hello 1 from parent', mypid #打印到输出流parentStdout, 经管道发送到子进程childStdin

    sys.stdout.flush()

    reply = raw_input()

    sys.stderr.write('Parent got: "%s"\n' % reply)#stderr没有绑定到管道上
    print 'Hello 2 from parent', mypid

    sys.stdout.flush()

    reply = sys.stdin.readline()#另外一种方式获得子进程返回信息

    sys.stderr.write('Parent got: "%s"\n' % reply[:-1])

pipetest.py代码如下:

#coding=utf-8

import os, time, sys
mypid = os.getpid()

parentpid = os.getppid()

sys.stderr.write('child %d of %d got arg: "%s"\n' %(mypid, parentpid, sys.argv[1]))
for i in range(2):

    time.sleep(3)

    recv = raw_input()#从管道获取数据,来源于父经常stdout

    time.sleep(3)

    send = 'Child %d got: [%s]' % (mypid, recv)

    print(send)#stdout绑定到管道上,发送到父进程stdin

    sys.stdout.flush()

输出:

child 7265 of 7264 got arg: "spam"

Parent got: "Child 7265 got: [Hello 1 from parent 7264]"

Parent got: "Child 7265 got: [Hello 2 from parent 7264]"
Python 相关文章推荐
python实现多线程采集的2个代码例子
Jul 07 Python
Python用UUID库生成唯一ID的方法示例
Dec 15 Python
python3中bytes和string之间的互相转换
Feb 09 Python
Python工程师面试必备25条知识点
Jan 17 Python
tensorflow训练中出现nan问题的解决
Feb 10 Python
django+mysql的使用示例
Nov 23 Python
python调用opencv实现猫脸检测功能
Jan 15 Python
Python3.5多进程原理与用法实例分析
Apr 05 Python
Python函数中参数是传递值还是引用详解
Jul 02 Python
使用APScheduler3.0.1 实现定时任务的方法
Jul 22 Python
Python 实现定积分与二重定积分的操作
May 26 Python
Python探索生命起源 matplotlib细胞自动机动画演示
Apr 21 Python
Python multiprocessing模块中的Pipe管道使用实例
Apr 11 #Python
Python httplib模块使用实例
Apr 11 #Python
初步探究Python程序的执行原理
Apr 11 #Python
Python与shell的3种交互方式介绍
Apr 11 #Python
Python函数参数类型*、**的区别
Apr 11 #Python
Python中的多重装饰器
Apr 11 #Python
Python中的各种装饰器详解
Apr 11 #Python
You might like
PHP中mb_convert_encoding与iconv函数的深入解析
2013/06/21 PHP
Laravel 5 框架入门(三)
2015/04/09 PHP
PHP获取当前文件的父目录方法汇总
2016/07/21 PHP
PHP读取word文档的方法分析【基于COM组件】
2017/08/01 PHP
Thinkphp5.0框架视图view的模板布局用法分析
2019/10/12 PHP
走出JavaScript初学困境—js初学
2008/12/29 Javascript
JavaScript this 深入理解
2009/07/30 Javascript
基于jQuery的实现简单的分页控件
2010/10/10 Javascript
Prototype源码浅析 String部分(二)
2012/01/16 Javascript
javascript倒计时功能实现代码
2012/06/07 Javascript
jquery validate poshytip 自定义样式
2012/11/26 Javascript
js鼠标点击事件在各个浏览器中的写法及Event对象属性介绍
2013/01/24 Javascript
js动态添加表格数据使用insertRow和insertCell实现
2014/05/22 Javascript
java、javascript实现附件下载示例
2014/08/14 Javascript
node.js中使用q.js实现api的promise化
2014/09/17 Javascript
JS实现同时搜索百度和必应的方法
2015/01/27 Javascript
Javascript进制转换实例分析
2015/05/14 Javascript
JS实现发送短信验证后按钮倒计时功能(防止刷新倒计时失效)
2017/07/07 Javascript
Vuex 使用 v-model 配合 state的方法
2018/11/13 Javascript
基于vue实现一个禅道主页拖拽效果
2019/05/27 Javascript
[01:58]最残酷竞争 2016国际邀请赛中国区预选赛积分循环赛回顾
2016/06/28 DOTA
实例Python处理XML文件的方法
2015/08/31 Python
python获取酷狗音乐top500的下载地址 MP3格式
2018/04/17 Python
在python中安装basemap的教程
2018/09/20 Python
django-rest-framework解析请求参数过程详解
2019/07/18 Python
Python xlrd excel文件操作代码实例
2020/03/10 Python
JetBrains PyCharm(Community版本)的下载、安装和初步使用图文教程详解
2020/03/19 Python
HTML5中的autofocus(自动聚焦)属性介绍
2014/04/23 HTML / CSS
北大研究生linux应用求职信
2013/10/29 职场文书
超市国庆节促销方案
2014/02/20 职场文书
党员2014两会学习心得体会
2014/03/17 职场文书
给老婆的检讨书1000字
2015/01/01 职场文书
Python Parser的用法
2021/05/12 Python
详解Java实践之建造者模式
2021/06/18 Java/Android
Win11运行育碧游戏总是崩溃怎么办 win11玩育碧游戏出现性能崩溃的解决办法
2022/04/06 数码科技
MySql分区类型及创建分区的方法
2022/04/13 MySQL