专栏首页随心DevOpsPython,Shell 和 三个标准文件

Python,Shell 和 三个标准文件

场景

使用 Python 执行 Shell 命令(或者脚本),有两种执行场景:

  1. 等待,直到命令执行完毕,一次性获取返回结果,做一些你想做的事情;
  2. 命令执行的同时,实时获取命令的持续输出,做一些你想做的事情。

例子

第一种场景:ls -a (list segment,Unix系统中使用非常频繁的命令)用于列出所有文件,文件列出之后就自动退出了。

第二种场景:ping zhihu.com 它会持续的输出结果,并不会退出(当然可以加 -t n 来指定 n 次之后结束退出,这样就属于场景一了,本文我们不指定 n)

相信这两种情况已满足了读者 90% 呃不 100% 需求了,如果没有请留言区留言


预备

执行一个 Shell 命令行时通常会自动打开三个标准文件,即:

  • 标准输入文件(stdin),通常对应终端的键盘
  • 标准输出文件(stdout)
  • 标准错误输出文件(stderr)

后两个文件对应终端的屏幕,进程从标准输入文件中得到输入数据,将正常输出数据输出到标准输出文件,而将错误信息送到标准错误文件中。

实战

比较一下比较好记忆:

ls -a (场景一)

import subprocess

def run_cmd(cmd):
    return subprocess.Popen(
        cmd,
        shell=False,
        stdout=subprocess.PIPE,
        stderr=subprocess.PIPE).communicate()

std_out_and_err = run_cmd(['ls', '-a'])
print(std_out_and_err)
# ('.\n..\ncontinuegetstdout.py\n', '')

两个点比较有价值:

  1. shell 这个参数很多人不理解,其实就是 False 的时候 Python 帮你执行命令, True 的时候相当于直接在终端执行命令。False 的时候,我们需要把命令按空格使用逗号分隔开来(即 list 数据结构)传给 cmd 参数(目的是让 Python 清楚这条命令的所有细节),代码中的例子就是使用这种;而 True 的时候只需要把命令一股脑(string 数据结构)的传给 cmd 参数,总结即 False:cmd=['ls', '-a'],True:cmd='ls -a' 纠结吗?推荐第一种
  2. communicate() 这个方法到底是干嘛的?官方文档如下:

Interact with process: Send data to stdin. Read data from stdout and stderr, until end-of-file is reached. Wait for process to terminate. The optional input argument should be a string to be sent to the child process, or None, if no data should be sent to the child. communicate() returns a tuple (stdout, stderr).

翻译一下:

与进程进行交互:将数据发送到 stdin。从 stdout 和 stderr 文件中读取数据,直到达到文件结尾。等待进程终止。可选的 input 参数应该是要发送到子进程的字符串,如果没有数据应该发送给子进程,则为 None。 communicate() 返回一个元组 (stdout, stderr)。

例子中我们并没有显式的指定 input 参数,默认为 None,我们只从 stdin 和 stderr 文件中读取数据。

这里其实是一个同步的过程,进程终止后才会返回所读取到的数据(进程终止,文件自然也结尾了),也就是 communicate() 方法直接使命令执行变为了同步,不执行完成就一直阻塞。如果我们执行 ping zhihu.com 等不会自己终止的命令,这种方式会一直卡死,因为进程一直没有结束,文件也一直不会结尾。

那么类似这种持续输出结果的命令如何执行呢?我们接着向下看

ping zhihu.com (场景二)

import subprocess

def run_cmd(cmd):
    return subprocess.Popen(
        cmd,
        shell=False,
        stdout=subprocess.PIPE,
        stderr=subprocess.PIPE)

p = run_cmd(['ping', 'zhihu.com'])
for i in iter(p.stdout.readline, ''):
    print(i.strip())

如代码所示,少了 communicate(),于是 p 是 subprocess.Popen 类的实例

查看 Popen 类源码 738 行:

self.stdout = os.fdopen(c2pread, 'rU', bufsize)

os.fdopen 的文档:

Return an open file object connected to a file descriptor.

得知 p.stdout 是一个:打开的文件对象,那么用 readline 就合理了。

注意这里使用了 iter() 内置函数,将 p.stdout 转换为一个迭代器,并使用 p.stdout.readline 替换迭代器的 next 方法,后面 '' 的意思就是当 p.stdout.readline 返回的值是 '' 的时候,迭代器终止,本篇不详细讲,有兴趣可以留言提问哦

本文分享自微信公众号 - 随心DevOps(heart-devops),作者:临书

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2018-01-21

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 你可能不知道的 Django Rest Framework 的两个新特性

    前端、甚至大前端发展如此火爆的 2017 年(我相信 2018 年一样)刚过去,像 MVC 架构那样直接使用后端模板渲染前端页面的方式已经不被推崇了(当然有些场...

    临书
  • 号称「永远不会输钱」的马丁格尔策略

    点进来,说明你想赢,那我们直入主题。 ---- 所谓「马丁格尔(Martingale)策略」是在某个赌盘里,当每次「输钱」时就以 2 的倍数再增加赌金,直到赢钱...

    临书
  • 你应该使用 Python 管理 Cron 作业

    在本教程中,您将了解 cron 作业的重要性以及为什么需要它们。你可以看一下 python-crontab,这是一个与 crontab 交互的 Python 模...

    临书
  • Class.getSimpleName()/Class/Class.getClass/Class.getClass.getName()

    qubianzhong
  • NVIDIA论坛常见Jetson问题汇总(2)

    (网址:https://docs.nvidia.com/jetson/archives/l4t-archived/l4t-3231/index.html#pag...

    GPUS Lady
  • C++重要知识点小结---2

    C++重要知识点小结--1 :http://www.cnblogs.com/heyonggang/p/3246631.html 1.C++允许程序员声明一个不能...

    猿人谷
  • C++重要知识点小结---2

    C++重要知识点小结--1 :http://www.cnblogs.com/heyonggang/p/3246631.html 1.C++允许程序员声明一个不能...

    猿人谷
  • 朴素贝叶斯分类器:例子解释

    在昨天推送了用一个例子引入贝叶斯公式的基本思想,然后用贝叶斯公式对一个很简单的问题做分类,最后引出来一个问题:后验概率 P(c | x) 的求解转化为求解 P(...

    double
  • 谈谈架构层级的“开闭原则”

    本文是关于架构层级SOLID原则的文章系列的第一篇。你可能熟悉如何在面向对象的层级遵循SOLID原则来进行类的设计,或者你也曾经疑惑这些原则是否适用于系统的架构...

    yuanyi928
  • [LeetCode] 39.Combination Sum

    【原题】 Given a set of candidate numbers (C) (without duplicates) and a target ...

    用户1148830

扫码关注云+社区

领取腾讯云代金券