zoukankan      html  css  js  c++  java
  • Python下调用Linux的Shell命令

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

    1. os 模块

    1.1. os模块的exec方法族

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


    1.2. os模块的system方法

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

    [python] view plaincopy在CODE上查看代码片派生到我的代码片
     
    1. >>> import os  
    2. >>> os.system("echo "Hello World"")   # 直接使用os.system调用一个echo命令  
    3. Hello World         ——————> 打印命令结果  
    4. 0                   ——————> What's this ? 返回值?  
    5. >>> val = os.system("ls -al | grep "log" ")   # 使用val接收返回值  
    6. -rw-r--r--  1 root       root       6030829 Dec 31 15:14 log    ——————> 此时只打印了命令结果  
    7. >>> print val             
    8. 0                   ——————> 注意,此时命令正常运行时,返回值是0  
    9. >>> val = os.system("ls -al | grep "log1" ")  
    10. >>> print val         
    11. 256                 ——————> 使用os.system调用一个没有返回结果的命令,返回值为256~  
    12. >>>   


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

    1.3. os模块的popen方法

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

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



    2. commands 模块

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

    *   commands.getstatusoutput(cmd)         返回(status, output)
    *   commands.getoutput(cmd)                   只返回输出结果
    *   commands.getstatus(file)                     返回ls -ld file的执行结果字符串,调用了getoutput,不建议使用此方法

    [python] view plaincopy在CODE上查看代码片派生到我的代码片
     
    1. long@zhouyl:/tmp/tests$ python  
    2. Python 2.7.3 (default, Jan  2013, 16:53:07)   
    3. [GCC 4.7.2] on linux2  
    4. Type "help", "copyright", "credits" or "license" for more information.  
    5. >>> import commands  
    6. >>> commands.getstatusoutput('ls -lt')      # 返回(status, output)  
    7. (0, 'total 5900 -rwxr-xr-x 1 long long      23 Jan  5 21:34 hello.sh -rw-r--r-- 1 long long     147 Jan  5 21:34 Makefile -rw-r--r-- 1 long long 6030829 Jan  5 21:34 log')  
    8. >>> commands.getoutput('ls -lt')            # 返回命令的输出结果(貌似和Shell命令的输出格式不同哈~)  
    9. 'total 5900 -rwxr-xr-x 1 long long      23 Jan  5 21:34 hello.sh -rw-r--r-- 1 long long     147 Jan  5 21:34 Makefile -rw-r--r-- 1 long long 6030829 Jan  5 21:34 log'  
    10. >>> commands.getstatus('log')               # 调用commands.getoutput中的命令对'log'文件进行相同的操作  
    11. '-rw-r--r-- 1 long long 6030829 Jan  5 21:34 log'  
    12. >>>   



    3. subprocess模块


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

    [python] view plaincopy在CODE上查看代码片派生到我的代码片
     
    1. >>> from subprocess import call  
    2. >>> call(["ls", "-l"])  
    3. >>> call('svn list svn://172.16.10.43/hossv2/haowu', shell=True)
    4. haowu-common/
    5. 0


    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会返回一个类文件对象,使用它来访问标准输入、输出。

    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可能更加适合你。

  • 相关阅读:
    [718. 最长重复子数组]
    排序算法--归并,堆,快速排序
    改进的插排--希尔排序
    排序算法--选泡插
    对封装继承多态的理解
    Servlet[springmvc]的Servlet.init()引发异常
    [面试题 16.18. 模式匹配]
    [124. 二叉树中的最大路径和](
    7.29_python_lx_day11
    7.28_python_lx_day18
  • 原文地址:https://www.cnblogs.com/hushaojun/p/4595235.html
Copyright © 2011-2022 走看看