Python下调用Linux的Shell命令的方法


Posted in Python onJune 12, 2018

有时候难免需要直接调用Shell命令来完成一些比较简单的操作,比如mount一个文件系统之类的。那么我们使用Python如何调用Linux的Shell命令?下面来介绍几种常用的方法:

1. os 模块

1.1. os模块的exec方法族

Python的exec系统方法同Unix的exec系统调用是一致的。这些方法适用于在子进程中调用外部程序的情况,因为外部程序会替换当前进程的代码,不会返回。( 这个看了点 help(os)  --> search "exec" 的相关介绍,但是没太搞明白咋使用)

1.2. os模块的system方法

system方法会创建子进程运行外部程序,方法只返回外部程序的运行结果。这个方法比较适用于外部程序没有输出结果的情况。

>>> import os 
>>> os.system("echo \"Hello World\"") # 直接使用os.system调用一个echo命令 
Hello World   ——————> 打印命令结果 
0     ——————> What's this ? 返回值? 
>>> val = os.system("ls -al | grep \"log\" ") # 使用val接收返回值 
-rw-r--r-- 1 root  root  6030829 Dec 31 15:14 log ——————> 此时只打印了命令结果 
>>> print val    
0     ——————> 注意,此时命令正常运行时,返回值是0 
>>> val = os.system("ls -al | grep \"log1\" ") 
>>> print val   
256     ——————> 使用os.system调用一个没有返回结果的命令,返回值为256~ 
>>>

注意:上面说了,此方法脂肪会外部程序的结果,也就是os.system的结果,所以如果你想接收命令的返回值,接着向下看~

1.3. os模块的popen方法

当需要得到外部程序的输出结果时,本方法非常有用。比如使用urllib调用Web API时,需要对得到的数据进行处理。os.popen(cmd) 要得到命令的输出内容,只需再调用下read()或readlines()等 如a=os.popen(cmd).read()

>>> os.popen('ls -lt')     # 调用os.popen(cmd)并不能得到我们想要的结果 
<open file 'ls -lt ', mode 'r' at 0xb7585ee8> 
>>> print os.popen('ls -lt').read()  # 调用read()方法可以得到命令的结果 
total 6064 
-rwxr-xr-x 1 long  long   23 Jan 5 21:00 hello.sh 
-rw-r--r-- 1 long  long   147 Jan 5 20:26 Makefile 
drwxr-xr-x 3 long  long   4096 Jan 2 19:37 test 
-rw-r--r-- 1 root  root  6030829 Dec 31 15:14 log 
drwxr-xr-x 2 long  long   4096 Dec 28 09:36 pip_build_long 
drwx------ 2 Debian-gdm Debian-gdm 4096 Dec 23 19:08 pulse-gylJ5EL24GU9 
drwx------ 2 long  long   4096 Jan 1 1970 orbit-long 
>>> val = os.popen('ls -lt').read()  # 使用变量可以接收命令返回值 
>>> if "log" in val:     # 我们可以使用in来判断返回值中有木有一个字符串 
...  print "Haha,there is the log" 
... else: 
...  print "No,not happy" 
... 
Haha,there is the log

2. commands 模块

使用commands模块的getoutput方法,这种方法同popend的区别在于popen返回的是一个文件句柄,而本方法将外部程序的输出结果当作字符串返回,很多情况下用起来要更方便些。
主要方法: 

  1. commands.getstatusoutput(cmd)         返回(status, output)
  2. commands.getoutput(cmd)                   只返回输出结果
  3. commands.getstatus(file)                     返回ls -ld file的执行结果字符串,调用了getoutput,不建议使用此方法
long@zhouyl:/tmp/tests$ python 
Python 2.7.3 (default, Jan 2 2013, 16:53:07) 
[GCC 4.7.2] on linux2 
Type "help", "copyright", "credits" or "license" for more information. 
>>> import commands 
>>> commands.getstatusoutput('ls -lt')  # 返回(status, output) 
(0, 'total 5900\n-rwxr-xr-x 1 long long  23 Jan 5 21:34 hello.sh\n-rw-r--r-- 1 long long  147 Jan 5 21:34 Makefile\n-rw-r--r-- 1 long long 6030829 Jan 5 21:34 log') 
>>> commands.getoutput('ls -lt')   # 返回命令的输出结果(貌似和Shell命令的输出格式不同哈~) 
'total 5900\n-rwxr-xr-x 1 long long  23 Jan 5 21:34 hello.sh\n-rw-r--r-- 1 long long  147 Jan 5 21:34 Makefile\n-rw-r--r-- 1 long long 6030829 Jan 5 21:34 log' 
>>> commands.getstatus('log')    # 调用commands.getoutput中的命令对'log'文件进行相同的操作 
'-rw-r--r-- 1 long long 6030829 Jan 5 21:34 log' 
>>>

3. subprocess模块

根据Python官方文档说明,subprocess模块用于取代上面这些模块。有一个用Python实现的并行ssh工具—mssh,代码很简短,不过很有意思,它在线程中调用subprocess启动子进程来干活。

>>> from subprocess import call 
>>> call(["ls", "-l"])

subprocess与system相比的优势是它更灵活(你可以得到标准输出,标准错误,“真正”的状态代码,更好的错误处理,等..)。我认为使用os.system已过时,或即将过时。

4. 众方法的比较以及总结

4.1. 关于 os.system

os.system("some_command with args")将命令以及参数传递给你的系统shell,这很好,因为你可以用这种方法同时运行多个命令并且可以设置管道以及输入输出重定向。比如:
os.system("some_command < input_file | another_command > output_file")
然而,虽然这很方便,但是你需要手动处理shell字符的转义,比如空格等。此外,这也只能让你运行简单的shell命令而且不能运行外部程序。

4.2. 关于os.popen

使用stream = os.popen("some_command with args")也能做与os.system一样的事,与os.system不同的是os.popen会给你一个像文件的对象从而你可以使用它来访问哪个程序的标准输入、输出。而且popen还有三个变种都是在I/O处理上有轻微不同。假如你通过一个字符串传递所有东西,你的命令会传递给shell;如果你通过一个列表传递他们,你不用担心逃避任何事。

4.3. 关于subprocess.popen

subprocess模块的Popen类,意图作为os.popen的替代,但是因为其很全面所以比os.popen要显得稍微复杂,使用起来需要学习哦~~。
比如你可以使用  print Popen("echo Hello World", stdout=PIPE, shell=True).stdout.read()  来替代  print os.popen("echo Hello World").read()。但是相比之下它使用一个统一的类包括4中不同的popen函数还是不错的。

4.4. 关于subprocess.call

subprocess模块的call函数。它基本上就像Popen类并都使用相同的参数,但是它只简单的等待命令完成并给你返回代码。比

如:

return_code = subprocess.call("echo Hello World", shell=True)

os模块中还有C中那样的fork/exec/spawn函数,但是我不建议直接使用它们。subprocess可能更加适合你。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
在SAE上部署Python的Django框架的一些问题汇总
May 30 Python
关于Python元祖,列表,字典,集合的比较
Jan 06 Python
Mac中升级Python2.7到Python3.5步骤详解
Apr 27 Python
python 实现一个贴吧图片爬虫的示例
Oct 12 Python
kaggle+mnist实现手写字体识别
Jul 26 Python
Python读取txt内容写入xls格式excel中的方法
Oct 11 Python
Python3 读取Word文件方式
Feb 13 Python
自定义Django默认的sitemap站点地图样式
Mar 04 Python
python实现excel公式格式化的示例代码
Dec 23 Python
详解Python中openpyxl模块基本用法
Feb 23 Python
Matplotlib可视化之添加让统计图变得简单易懂的注释
Jun 11 Python
详解Python类和对象内容
Jun 22 Python
Centos 升级到python3后pip 无法使用的解决方法
Jun 12 #Python
解决python升级引起的pip执行错误的问题
Jun 12 #Python
基于windows下pip安装python模块时报错总结
Jun 12 #Python
完美解决Python 2.7不能正常使用pip install的问题
Jun 12 #Python
无法使用pip命令安装python第三方库的原因及解决方法
Jun 12 #Python
pip命令无法使用的解决方法
Jun 12 #Python
解决Python安装后pip不能用的问题
Jun 12 #Python
You might like
PHP正则的Unknown Modifier错误解决方法
2010/03/02 PHP
PHP面向对象之工作单元(实例讲解)
2017/06/26 PHP
php生成条形码的图片的实例详解
2017/09/13 PHP
PHP中检查isset()和!empty()函数的必要性
2019/02/13 PHP
javascript动态向网页中添加表格实现代码
2014/02/19 Javascript
使用jquery实现的一个图片延迟加载插件(含图片延迟加载原理)
2014/06/05 Javascript
Javascript节点关系实例分析
2015/05/15 Javascript
详解javascript函数的参数
2015/11/10 Javascript
基于JavaScript实现网页倒计时自动跳转代码
2015/12/28 Javascript
多功能jQuery树插件zTree实现权限列表简单实例
2016/07/12 Javascript
javascript 分号总结及详细介绍
2016/09/24 Javascript
微信小程序 转发功能的实现
2017/08/04 Javascript
vue实现前进刷新后退不刷新效果
2018/01/26 Javascript
Angular 4.x+Ionic3踩坑之Ionic3.x pop反向传值详解
2018/03/13 Javascript
js canvas实现橡皮擦效果
2018/12/20 Javascript
vue - vue.config.js中devServer配置方式
2019/10/30 Javascript
微信小程序定义和调用全局变量globalData的实现
2019/11/01 Javascript
JS动态图片的实现方法完整示例
2020/01/13 Javascript
简单了解three.js 着色器材质
2020/08/03 Javascript
Vue实现鼠标经过文字显示悬浮框效果的示例代码
2020/10/14 Javascript
NodeJS和浏览器中this关键字的不同之处
2021/03/03 NodeJs
[14:00]DOTA2国际邀请赛史上最长大战 赛后专访B神
2013/08/10 DOTA
使用python调用浏览器并打开一个网址的例子
2014/06/05 Python
python连接mysql实例分享
2016/10/09 Python
Python实现多线程抓取网页功能实例详解
2017/06/08 Python
python实现读Excel写入.txt的方法
2018/04/29 Python
Python干货:分享Python绘制六种可视化图表
2018/08/27 Python
python中利用numpy.array()实现俩个数值列表的对应相加方法
2019/08/26 Python
使用OpenCV实现道路车辆计数的使用方法
2020/07/15 Python
Python实现手势识别
2020/10/21 Python
寻找迷宫的一条出路,o通路;X:障碍
2016/07/10 面试题
市场营销策划方案
2014/06/11 职场文书
复兴之路纪录片观后感
2015/06/02 职场文书
资产移交协议书
2016/03/24 职场文书
街道办残联2016年助残日活动总结
2016/04/01 职场文书
java项目构建Gradle的使用教程
2022/03/24 Java/Android