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 相关文章推荐
使用Python的Flask框架实现视频的流媒体传输
Mar 31 Python
将Python代码嵌入C++程序进行编写的实例
Jul 31 Python
Python中使用haystack实现django全文检索搜索引擎功能
Aug 26 Python
用Python登录好友QQ空间点赞的示例代码
Nov 04 Python
Python实现的特征提取操作示例
Dec 03 Python
Python3.7 新特性之dataclass装饰器
May 27 Python
基于h5py的使用及数据封装代码
Dec 26 Python
学会python自动收发邮件 代替你问候女友
May 20 Python
解决python运行启动报错问题
Jun 01 Python
Python API 操作Hadoop hdfs详解
Jun 06 Python
python 获取字典特定值对应的键的实现
Sep 29 Python
python BeautifulSoup库的安装与使用
Dec 17 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多个字符串替换成同一个的解决方法
2013/06/18 PHP
PHP中对缓冲区的控制实现代码
2013/09/29 PHP
php实现通用的信用卡验证类
2015/03/24 PHP
PHP实现的简单缓存类
2015/07/29 PHP
jquery 与NVelocity 产生冲突的解决方法
2011/06/13 Javascript
js加密解密字符串可自定义密码因子
2014/05/13 Javascript
JQuery分屏指示器图片轮换效果实例
2015/05/21 Javascript
使用JS代码实现点击按钮下载文件
2016/11/12 Javascript
webuploader模态框ueditor显示问题解决方法
2016/12/27 Javascript
JavaWeb表单及时验证功能在输入后立即验证(含用户类型,性别,爱好...的验证)
2017/06/09 Javascript
基于vue开发的在线付费课程应用过程
2018/01/25 Javascript
解决jquery的ajax调取后端数据成功却渲染失败的问题
2018/08/08 jQuery
JQuery中的常用事件、对象属性与使用方法分析
2019/12/23 jQuery
Vue微信公众号网页分享的示例代码
2020/05/28 Javascript
[03:42]2014DOTA2国际邀请赛 第三日比赛排位扑朔迷离
2014/07/12 DOTA
[42:24]完美世界DOTA2联赛循环赛 LBZS vs DM BO2第一场 11.01
2020/11/02 DOTA
Windows和Linux下使用Python访问SqlServer的方法介绍
2015/03/10 Python
python实现复制整个目录的方法
2015/05/12 Python
Python 判断是否为质数或素数的实例
2017/10/30 Python
火车票抢票python代码公开揭秘!
2018/03/08 Python
详解Python的循环结构知识点
2019/05/20 Python
django解决订单并发问题【推荐】
2019/07/31 Python
Series和DataFrame使用简单入门
2019/11/13 Python
200行python代码实现贪吃蛇游戏
2020/04/24 Python
用ldap作为django后端用户登录验证的实现
2020/12/07 Python
Pytorch实现WGAN用于动漫头像生成
2021/03/04 Python
HTML5新增的8类INPUT输入类型介绍
2015/07/06 HTML / CSS
理肤泉加拿大官网:La Roche-Posay加拿大
2018/07/06 全球购物
Nordgreen手表德国官方网站:丹麦极简主义手表
2019/10/31 全球购物
马云的职业生涯规划之路
2014/01/01 职场文书
课程改革实施方案
2014/03/16 职场文书
销售队伍口号
2014/06/11 职场文书
2014年旅游局法制宣传日活动总结
2014/11/01 职场文书
离婚财产分割协议书
2015/08/11 职场文书
初中班主任工作随笔
2015/08/15 职场文书
生日寿星公答谢词
2015/09/29 职场文书