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和Ruby中each循环引用变量问题(一个隐秘BUG?)
Jun 04 Python
python去除所有html标签的方法
May 05 Python
Linux下使用python自动修改本机网关代码分享
May 21 Python
python控制台中实现进度条功能
Nov 10 Python
Python+matplotlib实现计算两个信号的交叉谱密度实例
Jan 08 Python
Python使用pyodbc访问数据库操作方法详解
Jul 05 Python
Python将主机名转换为IP地址的方法
Aug 14 Python
pycharm双击无响应(打不开问题解决办法)
Jan 10 Python
如何利用Python 进行边缘检测
Oct 14 Python
python 利用openpyxl读取Excel表格中指定的行或列教程
Feb 06 Python
Python OpenCV超详细讲解基本功能
Apr 02 Python
Python字符串的转义字符
Apr 07 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
无线电的诞生过程
2021/03/01 无线电
SESSION信息保存在哪个文件目录下以及能够用来保存什么类型的数据
2012/06/17 PHP
JQuery Tips(3) 关于$()包装集内元素的改变
2009/12/14 Javascript
js DOM模型操作
2009/12/28 Javascript
jQuery 学习入门篇附实例代码
2010/03/16 Javascript
Jquery操作js数组及对象示例代码
2014/05/11 Javascript
JS实现可缩放、拖动、关闭和最小化的浮动窗口完整实例
2015/03/04 Javascript
jQuery+Ajax实现无刷新分页
2015/10/30 Javascript
JavaScript实现带播放列表的音乐播放器实例分享
2016/03/07 Javascript
Nodejs搭建wss服务器教程
2017/05/24 NodeJs
微信小程序开发之好友列表字母列表跳转对应位置
2017/09/26 Javascript
微信小程序实现定位及到指定位置导航的示例代码
2019/08/20 Javascript
深入理解令牌认证机制(token)
2019/08/22 Javascript
js实现翻牌小游戏
2020/07/31 Javascript
如何在 Vue 中使用 JSX
2021/02/14 Vue.js
解决python写的windows服务不能启动的问题
2014/04/15 Python
老生常谈Python基础之字符编码
2017/06/14 Python
Python入门之三角函数sin()函数实例详解
2017/11/08 Python
Django admin实现图书管理系统菜鸟级教程完整实例
2017/12/12 Python
对Xpath 获取子标签下所有文本的方法详解
2019/01/02 Python
10招!看骨灰级Pythoner玩转Python的方法
2019/04/15 Python
python文件读写代码实例
2019/10/21 Python
K近邻法(KNN)相关知识总结以及如何用python实现
2021/01/28 Python
HTML5几个设计和修改的页面范例分享
2015/09/29 HTML / CSS
canvas线条的属性详解
2018/03/27 HTML / CSS
美国开幕式潮店:Opening Ceremony
2018/02/10 全球购物
德国领先的大尺码和超大尺码男装在线零售商:Bigtex
2019/06/22 全球购物
电子狗项圈:eDog Australia
2019/12/04 全球购物
为什么Runtime.exec(“ls”)没有任何输出?
2014/10/03 面试题
战友聚会邀请函
2014/01/18 职场文书
2014院党委领导班子及其成员群众路线对照检查材料思想汇报
2014/10/04 职场文书
工作疏忽检讨书500字
2014/10/26 职场文书
病危通知书样本
2015/04/17 职场文书
社区党支部公开承诺书
2015/04/29 职场文书
标准发言稿结尾
2019/07/18 职场文书
Android Flutter实现3D动画效果示例详解
2022/04/07 Java/Android