python subprocess 非阻塞_Python subprocess 使用的几个小问题

一般调用

我们使用subprocess.Popen进行如下的子进程调用:

import subprocess

cmd = ['ls']

process = subprocess.Popen(cmd, stdout=subprocess.PIPE)

for stdout_line in iter(process.stdout.readline, b''):

print(stdout_line)

process.stdout.close()

return_code = process.wait()

if return_code:

raise subprocess.CalledProcessError(return_code, cmd)

shell=True

但当在Windows平台下运行dir命令,报错:FileNotFoundError: [WinError 2] The system cannot find the file specified。

解决方法,添加shell=True参数。

原因是:dir不是单独的命令,其是CMD(Windows 的shell)的内置程序,所以加shell=True参数。

注意:仅在绝对必要时才应使用shell=True。

import subprocess

cmd = ['dir']

process = subprocess.Popen(cmd, stdout=subprocess.PIPE, shell=True)

for stdout_line in iter(process.stdout.readline, b''):

print(stdout_line)

实时读取子进程输出

有时,发现运行的子进程,上述代码不会即时打印输出,而是要等个几秒钟。

原因是:子进程所运行的程序没有对stdout进行flush,比如C语言的printf没有即时flush时,其打印的输出其实是在缓冲区中的,得等缓冲区满了才集体输出一次。

解决方法:子进程如何运行的是Python,比较简单,添加os.environ["PYTHONUNBUFFERED"] = "1"即可。

如果是在Linux平台运行一般程序,添加cmd = ["stdbuf", "-oL"] + cmd即可。

如何为子进程添加环境变量

通过env=参数。

注意:只传我们想添加的那个环境变量(比如LD_LIBRARY_PATH)是不行,子进程默认是使用很多父进程的环境变量,因此先copy父进程的环境变量,再添加我们需要的。

my_env = os.environ.copy()

my_env['LD_LIBRARY_PATH'] = '../libs/x64/'

cmd = ['command']

popen = subprocess.Popen(cmd,

stdout=subprocess.PIPE,

env=my_env)

子进程切换工作目录

有时,我们的子进程需要在特定的目录里运行。

解决:使用cwd=环境变量。

pwd = os.path.dirname(__file__)

cmd = ['command']

popen = subprocess.Popen(cmd,

stdout=subprocess.PIPE,

cwd=os.path.join(pwd, '..', 'bin'))

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值