• 执行系统命令,subprocess使用说明



    os.system('ls -l') #只执行命令,不能将结果赋予变量

    os.system('mkdir test')  #创建test目录

    files = os.popen('ls -l').readlines() #可以将执行结果赋予变量

    os.system('./l.sh')  #运行l.sh脚本

    os.popen('./l.sh').readlines() #运行l.sh脚本

    os.system('powershell.exe ./a.ps1') #运行powershell脚本(a.ps1脚本位于当前目录下)

    如果shell=Trueexecutable可以用于指定用哪个shell来执行(比如bash、csh、zsh等)。*nix下,默认是 /bin/sh,windows下,就是环境变量 COMSPEC的值。windows下,只有当你要执行的命令确实是shell内建命令(比如dircopy等)时,你才需要指定shell=True,而当你要执行一个基于命令行的批处理脚本的时候,不需要指定此项。

    import subprocess

    import subprocess
    subprocess.call(['dir','c:\'],shell=True) #运行dir c:,print 显示结果
    #打开mstsc
    subprocess.Popen(['mstsc'],shell=True)  #如果使用call的话,则后面语句无法继续执行,除非先关闭mstsc
    调用变量:
    a='d:\'
    subprocess.call(['dir',a],shell=True)
    child = subprocess.Popen(['powershell', 'start-sleep', '-s', '3'],stdout=subprocess.PIPE, stderr=subprocess.PIPE)
    child.wait() #阻塞父进程,等待子进程结束再执行后面的语句
    print child.poll()
    print child.pid
    print 'parent process'
    import subprocess
    returncode = subprocess.call(['ping','www.baidu.com', '-n', '4'],shell=True) #ping www.baidu.com -n 4
    print returncode #返回 return code
    csvfile='c:\abcvm.csv'
    child = subprocess.Popen(['python.exe','./csv2json.py',csvfile],shell=True)  #带入参数csvfile
    #returncode = subprocess.Popen('ping www.baidu.com -n 4',shell=True) #不用中括号
    child.wait() #等待子进程执行完毕再执行父进程

     获取输出结果:

    import os,subprocess,sys
    reload(sys)
    sys.setdefaultencoding('utf-8') #在解码没有明确指明解码方式的时候使用
    
    child = subprocess.Popen(['ping', 'www.baidu.com', '-n', '2'],shell=True,stdout=subprocess.PIPE, stderr=subprocess.PIPE) #out和error都重定向到了PIPE对象中
    while child.poll() == None: #判断子进程是否结束
        c = child.stdout.readlines() 
        e = child.stderr.readlines() 
        for i in c:
            print i.strip().decode('GBK') #删掉空行、空格等字符,并对中文解码
    import os,subprocess,sys
    reload(sys)
    sys.setdefaultencoding('utf-8')
    
    child = subprocess.Popen(['ping', 'www.baidu.com', '-n', '2'], stdout=subprocess.PIPE, stderr=subprocess.PIPE)
    #communicate会阻塞父进程,直到子进程执行完成
    for i in child.communicate():
        print i.strip().decode('GBK') 

     stdin,stdout就是替换了管道符|,示例(netstat -an | findstr "3389"):

    import os,subprocess,sys
    c1 = subprocess.Popen(['netstat','-an'],shell=True,stdout=subprocess.PIPE)
    c2 = subprocess.Popen(['findstr','3389'],shell=True,stdin=c1.stdout,stdout=subprocess.PIPE)
    while c2.poll() == None:
        for i in c2.stdout.readlines():
            print i.strip()

     将结果输出到文本:

    f1 =open("e:\filew.txt","a")
    import os,subprocess,sys
    c1 = subprocess.Popen(['netstat','-an'],shell=True,stdout=subprocess.PIPE)
    c2 = subprocess.Popen(['findstr','3389'],shell=True,stdin=c1.stdout,stdout=subprocess.PIPE)
    while c2.poll() == None:
        for i in c2.stdout.readlines():
            print f1.write(i.strip() + '
    ')
    f1.close()

     

    ===========2016.5.5增加===========

    1. stdout直接执行cmd命令,如下等同于直接在cmd下执行 dir c: -L

    import subprocess
    a=subprocess.Popen(['dir','c:\'],shell=True,stdin=subprocess.PIPE,stdout=subprocess.PIPE)
    res = a.stdout.readlines()
    for i in res:
        print i.strip()

    2. stdin接收参数,直接执行cmd命令,如下等同于直接在cmd下执行 dir c: -L

    import subprocess
    a=subprocess.Popen(['dir'],shell=True,stdin=subprocess.PIPE,stdout=subprocess.PIPE)
    a.stdin.write('c:\ ')
    a.stdin.write('-L 
    ')
    res = a.communicate() #阻塞,等待子进程执行结束
    # print type(res) #tuple
    
    for i in res:
        print i

    3.stdin接收参数,先进入某一环境,再执行该环境下的命令,如在cmd下先进入python,然后再执行print 'aa'

    import subprocess
    a=subprocess.Popen(['python'],shell=True,stdin=subprocess.PIPE,stdout=subprocess.PIPE)
    a.stdin.write('print "aa"
    ')
    res = a.communicate()
    print res

    =================================

    从Python 2.4开始,Python引入subprocess模块来管理子进程,以取代一些旧模块的方法:如 os.system、os.spawn*、os.popen*、popen2.*、commands.*不但可以调用外部的命令作为子进程,而且可以连接到子进程的input/output/error管道,获取相关的返回信息。

    subprocess以及常用的封装函数

    运行python的时候,我们都是在创建并运行一个进程。像Linux进程那样,一个进程可以fork一个子进程,并让这个子进程exec另外一个程序。在Python中,我们通过标准库中的subprocess包来fork一个子进程,并运行一个外部的程序。

    subprocess包中定义有数个创建子进程的函数,这些函数分别以不同的方式创建子进程,所以我们可以根据需要来从中选取一个使用。另外subprocess还提供了一些管理标准流(standard stream)和管道(pipe)的工具,从而在进程间使用文本通信。

    subprocess.call()

    父进程等待子进程完成

    返回退出信息(returncode,相当于Linux exit code)

    subprocess.check_call()

    父进程等待子进程完成

    返回0

    检查退出信息,如果returncode不为0,则举出错误subprocess.CalledProcessError,该对象包含有returncode属性,可用try...except...来检查

    subprocess.check_output()

    父进程等待子进程完成

    返回子进程向标准输出的输出结果

    检查退出信息,如果returncode不为0,则举出错误subprocess.CalledProcessError,该对象包含有returncode属性和output属性,output属性为标准输出的输出结果,可用try...except...来检查。

    这三个函数的使用方法相类似,下面来以subprocess.call()举例说明:

     
    1. >>> import subprocess
    2. >>> retcode = subprocess.call(["ls", "-l"])
    3. #和shell中命令ls -a显示结果一样
    4. >>> print retcode
    5. 0

    将程序名(ls)和所带的参数(-l)一起放在一个表中传递给subprocess.call()

    shell默认为False,在Linux下,shell=False时, Popen调用os.execvp()执行args指定的程序;shell=True时,如果args是字符串,Popen直接调用系统的Shell来执行args指定的程序,如果args是一个序列,则args的第一项是定义程序命令字符串,其它项是调用系统Shell时的附加参数。

    上面例子也可以写成如下:

     
    1. >>> retcode = subprocess.call("ls -l",shell=True)

    在Windows下,不论shell的值如何,Popen调用CreateProcess()执行args指定的外部程序。如果args是一个序列,则先用list2cmdline()转化为字符串,但需要注意的是,并不是MS Windows下所有的程序都可以用list2cmdline来转化为命令行字符串。

    subprocess.Popen

    class Popen(args, bufsize=0, executable=None, stdin=None, stdout=None, stderr=None, preexec_fn=None, close_fds=False, shell=False, cwd=None, env=None, universal_newlines=False, startupinfo=None, creationflags=0)

    实际上,上面的几个函数都是基于Popen()的封装(wrapper)。这些封装的目的在于让我们容易使用子进程。当我们想要更个性化我们的需求的时候,就要转向Popen类,该类生成的对象用来代表子进程。

    与上面的封装不同,Popen对象创建后,主程序不会自动等待子进程完成。我们必须调用对象的wait()方法,父进程才会等待 (也就是阻塞block),举例:

     
    1. >>> import subprocess
    2. >>> child = subprocess.Popen(['ping','-c','4','blog.linuxeye.com'])
    3. >>> print 'parent process'

    从运行结果中看到,父进程在开启子进程之后并没有等待child的完成,而是直接运行print。

    对比等待的情况:

     
    1. >>> import subprocess
    2. >>> child = subprocess.Popen('ping -c4 blog.linuxeye.com',shell=True)
    3. >>> child.wait()
    4. >>> print 'parent process'

    从运行结果中看到,父进程在开启子进程之后并等待child的完成后,再运行print。

    此外,你还可以在父进程中对子进程进行其它操作,比如我们上面例子中的child对象:

     
    1. child.poll()           # 检查子进程状态
    2. child.kill()           # 终止子进程
    3. child.send_signal()    # 向子进程发送信号
    4. child.terminate()      # 终止子进程
    5. 子进程的PID存储在child.pid

    子进程的文本流控制

    子进程的标准输入、标准输出和标准错误如下属性分别表示:

    1. child.stdin
    2. child.stdout
    3. child.stderr

    可以在Popen()建立子进程的时候改变标准输入、标准输出和标准错误,并可以利用subprocess.PIPE将多个子进程的输入和输出连接在一起,构成管道(pipe),如下2个例子:

     
    1. >>> import subprocess
    2. >>> child1 = subprocess.Popen(["ls","-l"], stdout=subprocess.PIPE)
    3. >>> print child1.stdout.read(),

    #或者child1.communicate()

     
    1. >>> import subprocess
    2. >>> child1 = subprocess.Popen(["cat","/etc/passwd"], stdout=subprocess.PIPE)
    3. >>> child2 = subprocess.Popen(["grep","0:0"],stdin=child1.stdout, stdout=subprocess.PIPE)
    4. >>> out = child2.communicate()

    subprocess.PIPE实际上为文本流提供一个缓存区。child1的stdout将文本输出到缓存区,随后child2的stdin从该PIPE中将文本读取走。child2的输出文本也被存放在PIPE中,直到communicate()方法从PIPE中读取出PIPE中的文本。

    注意:communicate()是Popen对象的一个方法,该方法会阻塞父进程,直到子进程完成。

    from: https://blog.linuxeye.com/375.html

    http://www.cnblogs.com/xuxm2007/archive/2011/01/17/1937220.html

  • 相关阅读:
    CDH 2、Cloudera Manager的安装
    204 01 Android 零基础入门 03 Java常用工具类 04 Java集合 04 Map集合 01 Map概述
    203 01 Android 零基础入门 03 Java常用工具类 04 Java集合 03 Set集合 07 删除宠物猫信息数据(引入泛型知识点)
    202 01 Android 零基础入门 03 Java常用工具类 04 Java集合 03 Set集合 06 查找宠物猫信息数据
    201 01 Android 零基础入门 03 Java常用工具类 04 Java集合 03 Set集合 05 添加重复的宠物猫信息数据
    200 01 Android 零基础入门 03 Java常用工具类 04 Java集合 03 Set集合 04 添加和显式宠物猫信息
    199 01 Android 零基础入门 03 Java常用工具类 04 Java集合 03 Set集合 03 宠物猫信息管理概述
    198 01 Android 零基础入门 03 Java常用工具类 04 Java集合 03 Set集合 02 案例:在集合中插入字符串
    197 01 Android 零基础入门 03 Java常用工具类 04 Java集合 03 Set集合 01 Set概述
    196 01 Android 零基础入门 03 Java常用工具类 04 Java集合 02 List集合 05 案例:公告的删除和修改
  • 原文地址:https://www.cnblogs.com/dreamer-fish/p/5091066.html
Copyright © 2020-2023  润新知