工作中出于对一个脚本的优化,需要使用Python执行外部命令。这篇文章也算是工作的简单总结。
执行外部命令有多种方法可以实现。
1. os.system('command')
执行括号中的命令,执行成功返回“0”,执行失败返回“1”.
可见第一条执行成功返回0,第二条执行失败返回1,中间的字符为终端执行命令后返回的结果,不被python脚本捕获。
2. os.popen(command[, mode[, bufsize]])
command -- 使用的命令。
mode -- 模式权限可以是 'r'(默认) 或 'w'。
bufsize -- 指明了文件需要的缓冲大小:0意味着无缓冲;1意味着行缓冲;其它正值表示使用参数大小的缓冲(大概值,以字节为单位)。负的bufsize意味着使用系统的默认值,一般来说,对于tty设备,它是行缓冲;对于其它文件,它是全缓冲。如果没有改参数,使用系统的默认值。
popen() 创建一个管道,通过fork一个子进程,然后该子进程执行命令。返回值在标准IO流中,该管道用于父子进程间通信。父进程要么从管道读信息,要么向管道写信息,至于是读还是写取决于父进程调用popen时传递的参数(w或r)。
os.popen()也可以当作上下文管理器使用:
import os
with os.popen('echo "Hello"') as p:
print(p.read())
3. subprocess模块
3.1 subprocess.call(args, *, stdin= None, stdout = None, stderr = None, shell = False)
运行由args参数提供的命令,等待命令执行结束并返回返回码。args参数由字符串形式提供且有多个命令参数时,需要提供shell=True参数;如果把shell设置成True,指定的命令会在shell/cmd里解释执行。
返回值0或1,与os.system()完全相同。
3.2 subprocess.check_call(args, *, stdin = None, stdout = None, stderr = None, shell = False)
与call方法类似,不同在于如果命令行执行成功,check_call返回返回码0,否则抛出subprocess.CalledProcessError异常。
注意:不要为stdout和stderr参数赋值为subprocess.PIPE。
3.3 subprocess.check_output(args, *, stdin = None, stderr = None, shell = False, universal_newlines = False)
在子进程执行命令,以字符串形式返回执行结果的输出。如果子进程退出码不是0,抛出subprocess.CalledProcessError异常,异常的output字段包含错误输出。
可见命令执行结果返回到了test这个变量中,注意是二进制格式。
3.4 subprocess.Popen()
下面博客对具体参数等介绍的很详细,只想简单使用,可参考下面代码,博客介绍的很详细,就不在此赘述了。
# -*- coding: utf-8 -*-
import subprocess
a = subprocess.Popen('echo "Hello"', stdout=subprocess.PIPE, shell=True)
stdout, stderr = a.communicate()
print(stdout.decode('gbk').strip())
b = subprocess.Popen('echo "NiHao"', stdout=subprocess.PIPE, shell=True)
stdout2 = b.stdout.readline()
print(stdout2.decode('utf-8').strip())
参考博客:
python中的subprocess.Popen()使用
python中os.system、os.popen、subprocess.popen的区别