【module】subprocess

发布时间:2019-10-12 20:08:46编辑:auto阅读(2187)

    subprocess

    subprocess模块是python从2.4版本开始引入的模块。主要用来取代 一些旧的模块方法,如os.system、os.spawn、os.popen、commands.*等subprocess通过子进程来执行外部指令,并通过input/output/error管道,获取子进程的执行的返回信息。

    使用方法:

    运行外部命令:subprocess.call(command)

    subprocess的call方法可以用于执行一个外部命令,但该方法不能返回执行的结果,只能返回执行的状态码: 成功(0) 或 错误(非0)

    方法一:

    subprocess.call(['ls','-l'])
    总用量 4
    -rw-r--r-- 1 liangml liangml  0 3月   7 15:38 mark.md
    -rw-r--r-- 1 liangml liangml 68 3月  11 13:39 php-fpm.py
    0

    方法二:

    subprocess.call("ls -l",shell=True)
    总用量 4
    -rw-r--r-- 1 liangml liangml  0 3月   7 15:38 mark.md
    -rw-r--r-- 1 liangml liangml 68 3月  11 13:39 php-fpm.py
    0
    

    上面示例是在终端中运行,虽然可以看到运行结果,但实际取值只是状态码

    a = subprocess.call("ls -l",shell=True)
    总用量 4
    -rw-r--r-- 1 liangml liangml  0 3月   7 15:38 mark.md
    -rw-r--r-- 1 liangml liangml 81 3月  11 13:44 php-fpm.py
    print(a)
    0

    错误处理:subprocess.check_call()

    我们说过call执行返回一个状态码,我们可以通过check_call()函数来检测命令的执行结果,如果不成功将返回 subprocess.CalledProcessError 异常

    示例:

    import subprocess
    try:
        subprocess.check_call("sdf",shell=True)
    except subprocess.CalledProcessError as err:
        print("Command Error",err)

    捕获输出结果:subprocess.check_output()

    call()方法启动的进程,其标准输入输出会绑定到父进程的输入和输出。调用程序无法获取命令的输出结果。但可以通过check_output()方法来捕获输出。

    示例:

    output = subprocess.check_output("ls -l",shell=True)
    print(output.decode('utf-8'))
    import subprocess
    try:
        output = subprocess.check_output("lT -l",shell=True,stderr=subprocess.STDOUT)
    except subprocess.CalledProcessError as err:
        print("Command Error",err)
    #执行结果
    Command Error Command 'lT -l' returned non-zero exit status 127

    直接处理管道:subprocess.Popen()

    函数call(), check_call() 和 check_output() 都是Popen类的包装器。直接使用Popen会对如何运行命令以及如何处理其输入输出有更多控制。如通过为stdin, stdout和stderr传递不同的参数。

    • 与进程的单向通信:通过Popen()方法调用命令后执行的结果,可以设置stdout值为PIPE,再调用communicate()获取结果,返回结果为tuple. 在python3中结果为byte类型,要得到str类型需要decode转换一下

    输出结果(读)

    # 直接执行命令输出到屏幕
    >>> subprocess.Popen("ls -l",shell=True)
    <subprocess.Popen object at 0x7fe38cd67f98>
    总用量 4
    -rw-r--r-- 1 liangml liangml  0 3月   7 15:38 mark.md
    -rw-r--r-- 1 liangml liangml 76 3月  11 13:59 php-fpm.py
    # 不输出到屏幕,输出到变量
    >>> proc = subprocess.Popen(['echo','"Stdout"'],stdout=subprocess.PIPE)
    # communicate返回标准输出或标准出错信息
    >>> stdout_value = proc.communicate()
    >>> stdout_value
    (b'"Stdout"\n', None)
    
    
    
    >>> proc = subprocess.Popen(['ls','-l'],stdout=subprocess.PIPE)
    >>> stdout_value = proc.communicate()
    >>> stdout_value
    (b'\xe6\x80\xbb\xe7\x94\xa8\xe9\x87\x8f 48\ndrwxr-xr-x  3 liangml liangml 4096 3\xe6\x9c\x88  10 14:57 Desktop\ndrwxr-xr-x  6 liangml liangml 4096 3\xe6\x9c\x88   4 22:00 Documents\ndrwxr-xr-x  2 liangml liangml 4096 3\xe6\x9c\x88   8 18:32 Downloads\ndrwxr-xr-x  3 liangml liangml 4096 3\xe6\x9c\x88   3 11:10 GitBook\ndrwxr-xr-x  3 liangml liangml 4096 2\xe6\x9c\x88  22 17:52 Music\ndrwxr-xr-x  3 liangml liangml 4096 2\xe6\x9c\x88  22 17:48 Pictures\ndrwxr-xr-x  3 liangml liangml 4096 2\xe6\x9c\x88  26 18:29 PycharmProjects\ndrwxr-xr-x  3 liangml liangml 4096 2\xe6\x9c\x88  22 18:01 Steam\ndrwxr-xr-x 10 liangml liangml 4096 2\xe6\x9c\x88  27 11:21 svn\ndrwxr-xr-x  2 liangml liangml 4096 2\xe6\x9c\x88  22 17:48 Templates\ndrwxr-xr-x  2 liangml liangml 4096 2\xe6\x9c\x88  22 17:24 Videos\ndrwxr-xr-x  3 liangml liangml 4096 3\xe6\x9c\x88   7 21:00 vnote_notebooks\n', None)
    
    
    
    
    # 结果输出到文件
    >>> file_handle = open('/home/liangml/t.log','w+')
    >>> subprocess.Popen('ls -l',shell=True,stdout=file_handle)
    >>> subprocess.call("ls -l",shell=True)
    总用量 52
    -rw-r--r--  1 liangml liangml  779 3月  11 14:14 t.log
    vnote_notebooks

    与进程的双向通信

    proc = subprocess.Popen('cat', shell=True, stdin=subprocess.PIPE, stdout=subprocess.PIPE)
    msg = 'hello world'.encode('utf-8')
    # 写入到输入管道
    proc.stdin.write(msg)
    11  # 输入结果
    stdout_value = proc.communicate()
    stdout_value
    (b'hello world', None)  # 输出结果
    
    # 在需要进行相互交互的输入输出过程也可以使用shtin来实现
    # 以下实现打开python3的终端,执行一个print命令
    proc = subprocess.Popen(['python3'],stdin=subprocess.PIPE,stdout=subprocess.PIPE, stderr=subprocess.PIPE,)
    proc.stdin.write('print("helloworld")'.encode('utf-8'))
    out_value,err_value=proc.communicate()
    print(out_value)
    b'helloworld\n'
    print(err_value)
    b''

    Popen.communicate()方法用于和子进程交互:发送数据到stdin,并从stdout和stderr读数据,直到收到EOF。等待子进程结束。

    捕获错误输出

    proc = subprocess.Popen(['python3'],stdin=subprocess.PIPE,stdout=subprocess.PIPE, stderr=subprocess.PIPE,)
    proc.stdin.write('print "helloworld"'.encode('utf-8'))
    18
    out_value,err_value=proc.communicate()
    out_value
    b''
    print(err_value.decode('utf-8'))
      File "<stdin>", line 1
        print "helloworld"
                         ^
    SyntaxError: Missing parentheses in call to 'print'

    Popen其他方法

    • Popen.pid 查看子进程ID
    • Popen.returncode 获取子进程状态码,0表示子进程结束,None未结束

    在使用Popen调用系统命令式,建议使用communicate与stdin进行交互并获取输出(stdout),这样能保证子进程正常退出而避免出现僵尸进程:

    示例:

    proc = subprocess.Popen('ls -l', shell=True, stdout=subprocess.PIPE)
    # 当前子进程ID
    proc.pid
    5919
    # 返回状态None,进程未结束
    print(proc.returncode)
    None
    # 通过communicate提交后
    out_value = proc.communicate()
    proc.pid
    5919
    # 返回状态为0,子进程自动结束
    print(proc.returncode)
    0

关键字