3.7 执行外部命令

3.7.1 subprocess模块简介

subprocess模块是Python从2.4版本开始引入的模块。主要用来取代 一些旧的模块方法,如os.system、os.spawn*、os.popen*、commands.*等。subprocess通过子进程来执行外部指令,并通过input/output/error管道,获取子进程的执行的返回信息。

3.7.2 subprocess模块的主要函数

常用方法:

subprocess.run(args, *, stdin=None, input=None, stdout=None, stderr=None, shell=False, cwd=None, timeout=None, check=False, encoding=None, errors=None)

subprocess.run():执行命令,并返回执行状态,其中shell参数为False时,命令需要通过列表的方式传入,当shell为True时,可直接传入命令。

例如:其中注释掉的一句和下面的执行结果完全一样。

#!/usr/bin/python3
import subprocess

# print(subprocess.run(['df', '-hT'], shell=False))

print(subprocess.run('df -hT', shell=True))

结果为:

root@liu-ubuntu:~# ./14-sprun.py 
文件系统       类型      容量  已用  可用 已用% 挂载点
udev           devtmpfs  456M     0  456M    0% /dev
tmpfs          tmpfs      97M  4.9M   92M    6% /run
/dev/sda1      ext4       19G  5.0G   13G   28% /
tmpfs          tmpfs     482M  164K  482M    1% /dev/shm
tmpfs          tmpfs     5.0M     0  5.0M    0% /run/lock
tmpfs          tmpfs     482M     0  482M    0% /sys/fs/cgroup
tmpfs          tmpfs      97M   68K   97M    1% /run/user/1000
/dev/sr0       iso9660   1.6G  1.6G     0  100% /media/liu/Ubuntu 16.04.4 LTS amd64
tmpfs          tmpfs      97M     0   97M    0% /run/user/0
CompletedProcess(args='df -hT', returncode=0)

3.7.3 subprocess模块的Popen类

在一些复杂场景中,我们需要将一个进程的执行输出作为另一个进程的输入。在另一些场景中,我们需要先进入到某个输入环境,然后再执行一系列的指令等。这个时候我们就需要使用到suprocess的Popen()方法。

class subprocess.Popen(args, bufsize=-1, executable=None, stdin=None, stdout=None, stderr=None, preexec_fn=None, close_fds=True, shell=False, cwd=None, env=None, universal_newlines=False, startupinfo=None, creationflags=0, restore_signals=True, start_new_session=False, pass_fds=(), *, encoding=None, errors=None)

该方法有以下参数:

  • args:shell命令,可以是字符串,或者序列类型,如list,tuple;
  • bufsize:缓冲区大小,可不用关心;
  • stdin,stdout,stderr:分别表示程序的标准输入,标准输出及标准错误;
  • shell:与上面方法中用法相同;
  • cwd:用于设置子进程的当前目录;
  • env:用于指定子进程的环境变量。如果env=None,则默认从父进程继承环境变量;
  • universal_newlines:不同系统的的换行符不同,当该参数设定为true时,则表示使用\n作为换行符。

例如:

#!/usr/bin/python3
import subprocess

sp = subprocess.Popen('ifconfig', stdout=subprocess.PIPE, encoding='utf8')

cmd_out = sp.stdout.read()
sp.stdout.close()

print(cmd_out)

在此例中用到encoding这个参数,只在Python3.6以后有效,所以需要先升级Python3到3.6版本:

1.添加Python3.6安装包,并且安装
sudo apt-get install software-properties-common
sudo add-apt-repository ppa:jonathonf/python-3.6 
sudo apt-get update 
sudo apt-get install python3.6

2.修改系统默认Python3的版本为3.6
cd /user/bin 
rm python3 
ln -s python3.6 python3

3.验证python3的版本
root@liu-ubuntu:/usr/bin# python3
Python 3.6.5 (default, May  3 2018, 10:08:28) 
[GCC 5.4.0 20160609] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>>

然后上面程序运行结果为:

root@liu-ubuntu:~# ./15-sppopen.py 
ens32     Link encap:以太网  硬件地址 00:0c:29:6d:b9:c4  
          inet 地址:192.168.166.65  广播:192.168.166.255  掩码:255.255.255.0
          inet6 地址: fe80::e5c6:804b:12f1:c52c/64 Scope:Link
          UP BROADCAST RUNNING MULTICAST  MTU:1500  跃点数:1
          接收数据包:13881 错误:0 丢弃:0 过载:0 帧数:0
          发送数据包:11112 错误:0 丢弃:0 过载:0 载波:0
          碰撞:0 发送队列长度:1000 
          接收字节:14336412 (14.3 MB)  发送字节:1101974 (1.1 MB)

lo        Link encap:本地环回  
          inet 地址:127.0.0.1  掩码:255.0.0.0
          inet6 地址: ::1/128 Scope:Host
          UP LOOPBACK RUNNING  MTU:65536  跃点数:1
          接收数据包:286 错误:0 丢弃:0 过载:0 帧数:0
          发送数据包:286 错误:0 丢弃:0 过载:0 载波:0
          碰撞:0 发送队列长度:1000 
          接收字节:23269 (23.2 KB)  发送字节:23269 (23.2 KB)