使用subprocess.run()执行命令并捕获输出,推荐列表传参以避免注入风险;os.system()仅执行命令无输出捕获,os.popen()可读输出但已过时;错误处理可通过检查returncode、捕获stderr或使用try-except捕获CalledProcessError;后台执行用subprocess.Popen()并调用wait()等待结束;实时输出需结合Popen与TextIOWrapper逐行读取。

Python执行系统命令,本质上就是让Python程序能够调用操作系统提供的功能。方法有很多,关键在于选择最适合你需求的。
直接输出解决方案即可: Python提供了多种方法来执行系统命令,常见的有
os.system()、
os.popen()、
subprocess模块等。
subprocess模块功能最强大,也更推荐使用,因为它提供了更细粒度的控制和更安全的参数处理。
如何使用subprocess模块执行系统命令并获取输出?
subprocess模块的
run()函数是执行系统命令的首选方式。它可以捕获命令的输出、错误信息以及返回码。例如,要执行
ls -l命令并获取结果,可以这样做:
import subprocess result = subprocess.run(['ls', '-l'], capture_output=True, text=True, check=True) print(result.stdout)
这段代码中,
capture_output=True表示捕获命令的输出,
text=True表示将输出以文本形式返回,
check=True表示如果命令执行失败(返回码非0),则抛出异常。如果不想抛出异常,可以设置
check=False,然后检查
result.returncode。
subprocess.run()的第一个参数是一个列表,列表的每个元素都是命令及其参数。这样可以避免shell注入的风险。
立即学习“Python免费学习笔记(深入)”;
os.system()
和os.popen()
有什么区别,什么时候使用它们?
os.system()是最简单的执行系统命令的方式,但它功能有限,不推荐使用。它直接在shell中执行命令,返回命令的退出状态码。无法直接捕获命令的输出。
import os
os.system('ls -l')os.popen()可以执行系统命令并返回一个文件对象,可以读取命令的输出。但是,它也存在安全风险,并且在Python 3中已经被标记为过时。
import os
output = os.popen('ls -l').read()
print(output)通常情况下,
subprocess模块能够替代
os.system()和
os.popen()的所有功能,并且更加安全和灵活。只有在一些非常简单的场景下,或者需要兼容旧代码时,才可能考虑使用它们。
如何处理执行系统命令时可能出现的错误?
执行系统命令时,可能会遇到各种错误,例如命令不存在、权限不足、参数错误等。使用
subprocess模块时,可以通过以下方式处理这些错误:
检查返回码:
subprocess.run()
返回的result
对象包含returncode
属性,表示命令的退出状态码。如果returncode
非0,则表示命令执行失败。捕获标准错误:
subprocess.run()
的capture_output=True
可以同时捕获标准输出和标准错误。可以通过result.stderr
访问标准错误信息。
Shell脚本编写基础 中文WORD版下载Shell本身是一个用C语言编写的程序,它是用户使用Linux的桥梁。Shell既是一种命令语言,又是一种程序设计语言。作为命令语言,它交互式地解释和执行用户输入的命令;作为程序设计语言,它定义了各种变量和参数,并提供了许多在高级语言中才具有的控制结构,包括循环和分支。它虽然不是Linux系统核心的一部分,但它调用了系统核心的大部分功能来执行程序、建立文件并以并行的方式协调各个程序的运行。因此,对于用户来说,shell是最重要的实用程序,深入了解和熟练掌握shell的特性极其使用方法,是用好Linux系统
使用
try...except
块: 可以使用try...except
块捕获subprocess.CalledProcessError
异常。这个异常会在check=True
时,命令执行失败时抛出。
import subprocess
try:
result = subprocess.run(['ls', '-z'], capture_output=True, text=True, check=True)
print(result.stdout)
except subprocess.CalledProcessError as e:
print(f"命令执行失败:{e}")
print(f"错误信息:{e.stderr}")这段代码尝试执行一个不存在的命令
ls -z,由于
check=True,所以会抛出
subprocess.CalledProcessError异常,并在
except块中捕获并打印错误信息。
如何安全地传递参数给系统命令?
直接拼接字符串来构建命令是很危险的,因为它容易受到shell注入攻击。应该使用列表来传递命令及其参数,这样
subprocess模块会自动处理参数的转义和引用。
例如,假设要执行
grep命令,搜索包含空格的字符串:
import subprocess search_string = 'hello world' result = subprocess.run(['grep', search_string, 'file.txt'], capture_output=True, text=True) print(result.stdout)
如果直接使用字符串拼接,可能会出现问题,特别是当
search_string包含特殊字符时。使用列表可以避免这些问题。
如何在后台执行系统命令?
要在后台执行系统命令,可以使用
subprocess.Popen()函数。它会创建一个新的进程来执行命令,并且不会阻塞当前进程。
import subprocess
process = subprocess.Popen(['sleep', '10']) # 在后台执行sleep 10命令
# 可以执行其他操作,不会被sleep命令阻塞
print("命令已在后台执行")
process.wait() # 等待后台进程结束
print("命令执行完毕")subprocess.Popen()返回一个
Popen对象,可以使用它的
wait()方法等待后台进程结束。如果不调用
wait()方法,后台进程会继续运行,直到程序退出。
如何实时输出系统命令的执行结果?
有时候,需要实时显示系统命令的执行结果,而不是等到命令执行完毕后再输出。可以使用
subprocess.Popen()和
io.TextIOWrapper来实现这个功能。
import subprocess
import io
process = subprocess.Popen(['ping', 'www.google.com'], stdout=subprocess.PIPE, stderr=subprocess.PIPE)
stdout_reader = io.TextIOWrapper(process.stdout, encoding='utf-8')
stderr_reader = io.TextIOWrapper(process.stderr, encoding='utf-8')
while True:
stdout_line = stdout_reader.readline()
stderr_line = stderr_reader.readline()
if stdout_line:
print(stdout_line.strip())
if stderr_line:
print(stderr_line.strip())
if process.poll() is not None:
break
process.wait()这段代码使用
subprocess.Popen()执行
ping www.google.com命令,并将标准输出和标准错误重定向到管道。然后,使用
io.TextIOWrapper将管道转换为文本流,并实时读取和输出每一行。
process.poll()用于检查进程是否已经结束。









