【python】在Python代码中执行Linux命令的详细用法教程

news2025/1/11 7:55:35

在这里插入图片描述

✨✨ 欢迎大家来到景天科技苑✨✨

🎈🎈 养成好习惯,先赞后看哦~🎈🎈

🏆 作者简介:景天科技苑
🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。
🏆《博客》:Python全栈,PyQt5和Tkinter桌面开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi,flask等框架,云原生K8S,linux,shell脚本等实操经验,网站搭建,数据库等分享。

所属的专栏:python综合应用,基础语法到高阶实战教学
景天的主页:景天科技苑

在这里插入图片描述

文章目录

  • 在Python代码中执行Linux命令的详细用法教程
    • 一、使用`os`模块执行Linux命令
      • 1.1 `os.system()`
      • 1.2 `os.popen()`
    • 二、使用`subprocess`模块执行Linux命令
      • 2.1 `subprocess.run()`
      • 2.2 `subprocess.Popen()`
      • 2.3 `subprocess.call()`
    • 三、使用`sh`模块执行Linux命令
    • 四、安全注意事项
    • 五、进阶用法与技巧
      • 5.1 异步执行命令
      • 5.2 捕获命令的实时输出
      • 5.3 使用环境变量
    • 六、总结

在Python代码中执行Linux命令的详细用法教程

在Python开发过程中,经常需要执行Linux系统命令来完成各种任务,如文件操作、系统状态检查等。Python提供了多种方式来调用和执行系统命令,其中os模块和subprocess模块是最常用的两种。本文将详细介绍如何在Python代码中执行Linux命令,并结合实际案例来演示这些方法的使用。

一、使用os模块执行Linux命令

1.1 os.system()

os.system()函数是os模块中最直接执行系统命令的方式。它会执行指定的命令并等待命令执行完成,然后返回命令的退出状态码。退出状态码为0通常表示命令执行成功,非0值表示执行失败。

示例代码

import os

# 执行ls命令并打印结果
result = os.system('ls -l')
print(f"命令执行结果: {result}")

# 如果需要获取命令的输出,则此方法不适用,因为os.system()不直接返回命令的输出。

1.2 os.popen()

os.popen()函数可以执行命令并返回一个文件对象,你可以像操作文件一样读取命令的输出。这种方式比os.system()更灵活,因为它允许你获取命令的输出。

示例代码

import os

# 使用os.popen执行命令并读取输出
with os.popen('ls -l') as command:
    for line in command:
        print(line, end='')

# 注意:os.popen()在Python 3.x中已被标记为不推荐使用,推荐使用subprocess模块。

二、使用subprocess模块执行Linux命令

subprocess模块提供了更强大和灵活的执行系统命令的功能。它允许你创建新的进程,连接到它们的输入/输出/错误管道,并获取它们的返回码。

2.1 subprocess.run()

subprocess.run()是Python 3.5及以上版本中推荐使用的函数,用于执行命令并等待其完成。它返回一个CompletedProcess实例,其中包含命令的输出、错误输出和返回码。

示例代码

import subprocess

# 执行ls命令并捕获输出
result = subprocess.run(['ls', '-l'], capture_output=True, text=True)

# 检查命令是否成功执行
if result.returncode == 0:
    print(f"命令输出:\n{result.stdout}")
else:
    print(f"命令执行失败,错误输出:\n{result.stderr}")

2.2 subprocess.Popen()

subprocess.Popen()用于创建新的进程,并且不等待命令执行完成。它返回一个Popen对象,允许你与子进程进行交互。这种方式提供了更细粒度的控制,比如可以实时获取命令的输出。

示例代码

import subprocess

# 使用Popen执行命令并实时获取输出
process = subprocess.Popen(['ls', '-l'], stdout=subprocess.PIPE, text=True)

# 读取输出
for line in process.stdout:
    print(line, end='')

# 等待子进程完成
process.wait()

# 检查命令是否成功执行
if process.returncode == 0:
    print("命令执行成功")
else:
    print("命令执行失败")

2.3 subprocess.call()

subprocess.call()执行指定的命令,并等待命令执行完成。它返回命令的退出状态码,但不提供直接获取命令输出的方式。

示例代码

import subprocess

# 执行命令并获取退出状态码
result = subprocess.call(['ls', '-l'])

# 检查命令是否成功执行
if result == 0:
    print("命令执行成功")
else:
    print("命令执行失败")

# 注意:如果需要获取命令的输出,请使用subprocess.run()或subprocess.Popen()。

三、使用sh模块执行Linux命令

sh是一个第三方库,提供了一个更简洁和友好的方式来执行系统命令。它模仿了Shell的语法和行为,使得在Python中执行Shell命令变得更加容易。

安装sh模块

首先,你需要安装sh模块。可以通过pip来安装:

pip install sh

示例代码

import sh

# 创建一个ls命令对象
ls = sh.Command("ls")

# 执行命令,并将结果输出到终端
ls("-l")

# 或者,你可以捕获命令的输出
output = ls("-l", _out=sh.piping.PIPE)
print(output.stdout.decode('utf-8'))  # 注意解码输出,因为sh库默认返回bytes

# 使用sh库执行带参数的命令也很直观
grep = sh.Command("grep")
result = grep("some_text", "some_file.txt")

if result.exit_code == 0:
    print("找到匹配项:", result.stdout.decode('utf-8'))
else:
    print("未找到匹配项")

# sh库还提供了管道和重定向的支持
cat = sh.Command("cat")
sort = sh.Command("sort")

# 管道示例:将cat的输出传递给sort
sorted_output = (cat("input.txt") | sort)()
print(sorted_output.stdout.decode('utf-8'))

# 注意:在上面的管道示例中,我们使用了Python的管道操作符 `|`,但在 `sh` 库中是通过特殊方式实现的。
# 实际使用时,需要在括号内调用整个管道表达式,如上例所示。

四、安全注意事项

在Python中执行系统命令时,需要特别注意安全问题,特别是当命令或命令的参数来自不可信的源时。以下是一些安全建议:

  1. 避免命令注入:如果命令的某部分(如参数)来自用户输入,请确保不要直接将其插入到命令字符串中。使用列表形式(如subprocess.run(['ls', '-l']))而不是字符串形式(如subprocess.run('ls -l')),因为列表形式能够更好地处理包含空格、引号等特殊字符的参数。

  2. 限制权限:确保执行命令的进程没有不必要的权限。例如,如果命令不需要写入文件系统的权限,则不要以root用户身份运行Python脚本。

  3. 验证输入:在将输入用于命令之前,验证其是否符合预期格式和范围。

  4. 使用安全的库:对于复杂的需求,考虑使用专门为此设计的库,这些库可能已经解决了安全问题和边缘情况。

  5. 记录和监控:对执行的命令和结果进行记录和监控,以便在出现问题时能够进行回溯和调查。

五、进阶用法与技巧

5.1 异步执行命令

如果你需要在不阻塞主程序的情况下执行系统命令,可以考虑使用异步编程。Python的asyncio库可以与subprocess模块结合使用,通过asyncio.create_subprocess_exec()asyncio.create_subprocess_shell()函数来异步执行命令。

示例代码(使用asyncio.create_subprocess_exec()):

import asyncio

async def run_command(cmd, *args):
    # 创建子进程
    process = await asyncio.create_subprocess_exec(
        cmd, *args,
        stdout=asyncio.subprocess.PIPE,
        stderr=asyncio.subprocess.PIPE,
    )
    
    # 等待子进程完成
    stdout, stderr = await process.communicate()
    
    # 解码输出
    if stdout:
        print(f'[stdout]\n{stdout.decode()}')
    if stderr:
        print(f'[stderr]\n{stderr.decode()}')

# 异步执行ls命令
asyncio.run(run_command('ls', '-l'))

注意:在这个例子中,我们使用了asyncio.run()来运行异步函数,这是Python 3.7及以上版本中推荐的方式。

5.2 捕获命令的实时输出

如果你想要实时捕获命令的输出,而不是等待命令执行完成后再一次性获取输出,你可以使用subprocess.Popen()结合多线程或多进程来实现。但更简洁的方法是使用asyncio(如果适用)或threading模块中的Thread类,并设置stdout=subprocess.PIPE来读取输出。

然而,由于subprocess.Popen().stdout是一个阻塞的流,直接使用可能会遇到问题。一个常见的解决方案是使用io.TextIOWrapperreadline()方法,结合循环来逐行读取输出。

示例代码(使用threadingsubprocess.Popen()):

import subprocess
import threading

def read_output(proc):
    for line in iter(proc.stdout.readline, b''):
        print(line.decode('utf-8').strip())

# 创建并启动子进程
proc = subprocess.Popen(['tail', '-f', 'some_log_file.log'], stdout=subprocess.PIPE, text=True)

# 启动线程来读取输出
t = threading.Thread(target=read_output, args=(proc,))
t.start()

# 注意:这个例子中使用了tail -f来模拟实时输出。你可能需要其他方式来停止线程,
# 比如设置某个标志位,并在read_output函数中检查这个标志位来决定是否退出循环。

5.3 使用环境变量

有时,你可能需要在执行命令时设置或修改环境变量。subprocess.run()subprocess.Popen()都允许你通过env参数来指定一个环境变量字典。

示例代码

import subprocess

# 设置环境变量
env = os.environ.copy()
env["MY_VAR"] = "some_value"

# 执行命令,使用自定义的环境变量
result = subprocess.run(['echo', '$MY_VAR'], shell=True, env=env, text=True)
print(result.stdout)  # 注意:在shell=True时,环境变量替换由shell完成,而不是Python。

# 如果不想使用shell=True(出于安全考虑),你可能需要修改命令字符串来直接引用环境变量
# 但这通常不适用于动态设置的环境变量,因为命令字符串在Python中就已经被解析了。
# 对于这种情况,你可能需要寻找其他方法来将环境变量传递给命令。

注意:在上面的echo示例中,由于我们使用了shell=True,环境变量替换实际上是由shell完成的,而不是Python。这意呀着$MY_VAR会被shell替换为环境变量MY_VAR的值。然而,出于安全考虑,通常建议避免使用shell=True,除非你能确保传递给shell的命令是安全的。

六、总结

在Python中执行Linux命令是一个强大且灵活的功能,可以通过多种方法实现。os模块提供了基本的命令执行功能,但subprocess模块因其灵活性和强大功能而被广泛推荐。此外,第三方库如sh也提供了更加简洁和易于使用的接口。无论使用哪种方法,都需要注意安全问题,并避免命令注入等潜在风险。同时,通过结合异步编程和线程/进程管理等技术,你可以实现更加复杂和高效的命令执行逻辑。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1959883.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

昇思25天学习打卡营第01天|昇思MindSpore大模型基础介绍

昇思MindSpore和华为昇思MindSpore大模型学习打卡系列文章,本文仅供参考~ 文章目录 前言一、昇思MindSpore是什么?二、执行流程三、设计理念四、层次结构五、Huawei昇腾AI全栈 前言 随着计算机大模型的不断发展,Ai这门技术也越来越重要&#…

免费【2024】springboot 宠物领养救助平台的开发与设计

博主介绍:✌CSDN新星计划导师、Java领域优质创作者、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和学生毕业项目实战,高校老师/讲师/同行前辈交流✌ 技术范围:SpringBoot、Vue、SSM、HTML、Jsp、PHP、Nodejs、Python、爬虫、数据可视化…

Prozyme糖样本检测平台--GlykoPrep® Rapid N-Glycan Preparation with APTS

单克隆抗体已成为生物制药行业具有潜力的新兴蛋白候选药物。其药物研发流程包括一系列精细的控制和评估步骤,需要仔细、严格地监测目标化合物的治疗稳定性及有效性。因此,在商业化前的每个阶段对单克隆抗体进行全面表征是极其有益的。在大量研究成熟的蛋…

用Python打造精彩动画与视频,3.1 安装和设置 MoviePy

第3章 开始你的第一个视频项目 MoviePy 3.1 安装和设置 MoviePy 视频处理在多媒体内容创作中起着至关重要的作用。MoviePy 是一个用于视频编辑的 Python 库,它可以让你轻松地进行视频剪辑、合并、特效添加等操作。通过 MoviePy,你可以用简单的 Python…

基于Java+SpringBoot+Vue的的课程作业管理系统

前言 ✌全网粉丝20W,csdn特邀作者、博客专家、CSDN[新星计划]导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 🍅文末获取项目下载方式🍅 哈喽兄弟们,好久不见哦&#xff5…

IEEE 快刊合集!期刊选的好,JCR1区照样轻松拿下~

【SciencePub学术】今天小编给大家推荐2本计算机领域的SCI,均隶属于IEEE出版社,虽比不上前几日给大家介绍的IEEE-Trans系列的那本期刊优秀,但是放在行业内还是很拿得出手的。 现在提交,可免费预审,预审通过后录用率100…

失业负债女孩,下班后用AI做副业,快速翻身上岸

** - 我们应该把负债看成是成长的助力,而不是搞垮骆驼的稻草。 负债上岸,没有更多的捷径,唯有“开源节流”,节流就是尽可能节约花费,把生活支出减少,开源就是尽可能多的增加自己的副业收入。 负债后的至…

【通俗理解】“多即不同”观念的科学内涵——从对称破缺到科学层次结构的探索

【通俗理解】“多即不同”观念的科学内涵——从对称破缺到科学层次结构的探索 对称破缺与科学层次结构的类比 你可以把对称破缺比作一个“建筑师”,它构建了科学大厦的基础框架。而科学层次结构则是一个“城市规划师”,它规划了这些基础框架如何组合成一…

【附精彩文章合辑】跨界对话:黄仁勋与扎克伯格的“外套交换”与未来展望

跨界对话:黄仁勋与扎克伯格的“外套交换”与未来展望 在科技界的浩瀚星空中,两位璀璨的名字如同双子星般交相辉映——NVIDIA的创始人兼CEO黄仁勋(Jensen Huang)与Facebook(现更名为Meta)的掌舵人马克扎克伯…

快速开启react+electron应用,搭建启动问题

注意: React 本地启动在 3000端口Electron 在创建 BrowserWindow 的时候,可以读取本地的文件或者是 url开发环境 读取localhost: 3000生产环境 需要加载本地成型以后的本地文件,打包的时候再考虑 一 react 脚手架 create-react-app 快速搭建…

iTerm2使用手册

iTerm2字体 参考文章:iTerm2 安装 SF Mono 字体 iTerm2默认的字体不太好看,感觉Mac自带的终端的字体看起来很舒服,于是让iTerm2和Mac自带终端保持一致 找到Terminal.app下面的字体 执行 cp *.otf ~/Library/Fonts/ 然后在iTerm2中设置字…

ElementUI,el-input输入框max、min限制最大最小值失效

<el-input type"number" v-model"loadNodesNum" :min"1" :max"5" style"width: 240px;overflow: hidden;"><el-button slot"append" click"handleMaxLoadNum(1)">负载最多的节点TOP</el…

记一次JS到WAF绕过上传

一、前言 某渗透项目中和队友配合挖到一个通用系统漏洞&#xff0c;主要对JS接口进行分析&#xff0c;经历一波绕waf的曲折最终getshell&#xff0c;文笔粗劣&#xff0c;大佬勿喷。 二、JS分析 在看到某系统仅仅一个登陆框&#xff0c;一波弱口令尝试和目录扫描&#xff0c…

PHP教务排课系统小程序源码

教务排课系统&#xff1a;智慧教育的新篇章 &#x1f4da; 开篇&#xff1a;告别手动排课的繁琐时代 在这个信息爆炸的时代&#xff0c;教育领域的每一个细节都在向智能化迈进。你是否还记得&#xff0c;曾经老师们为了编排一张完美的课程表&#xff0c;需要花费多少时间和精…

JMeter接口测试-5.JMeter高级使用

JMeter高级使用 案例&#xff1a; 用户登录后-选择商品-添加购物车-创建订单-验证结果 问题&#xff1a; JMeter测试中&#xff0c;验证结果使用断言&#xff0c;但断言都是固定的内容假如要判断的内容(预期内容)是在变化的, 有时候还是不确定的, 那该怎么办呢? 解决&…

【MySQL】访问mysqld的方式{命令行客户端/vscode-c-api客户端/图形化界面:mysql/navicat}

文章目录 1.访问数据库2.vscode-c-api客户端方法一&#xff1a;重新安装相关数据方法二&#xff1a;直接使用之间安装mysql的附加数据查看mysql文档方式使用c-mysql-api库访问mysqld代码理解mysql_res 3.图形化界面 1.访问数据库 命令行式客户端向mysqld下达指令 网页版 图形化…

ViVim编辑器

文章目录 基本介绍常用的三种模式1. 正常模式2. 插入模式3. 命令行模式 常用快捷键环境参数配置光标移动查找与替换删除复制粘贴模式切换 基本介绍 Linux 系统会内置 vi 文本编辑器 Vi是最早的文本编辑器之一&#xff0c;主要用于Linux和Unix操作系统中的文本编辑。它是一个纯…

Kafka动态授权认证:利用SASL/SCRAM机制提升安全性

摘要 Apache Kafka是一个流行的分布式流处理平台&#xff0c;其安全性对于保护数据传输至关重要。SASL/SCRAM&#xff08;Simple Authentication and Security Layer/Salted Challenge Response Authentication Mechanism&#xff09;是一种认证机制&#xff0c;可以为Kafka集…

Spring-原理篇-DispatcherServlet 初始化 怎么和IOC进行了打通?

委托模式的体现&#xff0c;在初始化醒目的时候Spring MVC为我们提供了一个DispatcherServlet&#xff0c;映射了所有的路径&#xff0c;所有的请求都会先到达这里然后被转发到具体的Controller 进行处理&#xff0c;此文来探索一下&#xff0c;DispatcherServlet 初始化的时候…

强化学习学习(四)Model-Based Reinforcement Learning 从原理到改进分析

文章目录 Model-Based RLBasics of Model-based reinforcement learningUncertainty of Model-based RLUse uncertainty in Model-based RLState space (latent space) models Model-Based RL Basics of Model-based reinforcement learning Model-based reinforcement learn…