Python執(zhí)行外部命令subprocess的使用詳解
subprocess.run()方法是官方推薦的方法,幾乎所有的工作都可以用它來完成。如下是函數(shù)源碼:
subprocess.run(args, *,stdin=None,input=None,stdout=None,stderr=None,shell=False,ced=None,timeout=None,check=False,enccoding=None,error=None)
該函數(shù)返回一個(gè)CompletedProcess類(有屬性傳入?yún)?shù)及返回值)的實(shí)例,該函數(shù)的參數(shù)有很多,只需要記住常用的即可
1、args : 代表需要在操作系統(tǒng)中執(zhí)行的命令,可以是字符串形式(要求shell=True),也可以是list列表類型2、* :代表可變參數(shù),一般是列表或者字典類型3、stdin、stdout、stderr :指定了可執(zhí)行程序的標(biāo)準(zhǔn)輸入、標(biāo)準(zhǔn)輸出、標(biāo)準(zhǔn)錯(cuò)誤文件句柄4、shell :代表著程序是否需要在shell上執(zhí)行,當(dāng)想使用shell的特性時(shí),設(shè)置shell=True,這樣就可以使用shell指令的管道、文件名稱通配符、環(huán)境變量等,不過python也提供了很多類似shell的模塊,如glob、fnmatch、os.walk()、os.path.expandvars()、os.path.expanduser()和shutil。5、check :如果check設(shè)置為True,就檢查命令的返回值,當(dāng)返回值為0時(shí),將拋出AclledProcessError的異常6、timeout :設(shè)置超時(shí)時(shí)間,如果超時(shí)則kill掉子進(jìn)程
1.使用字符串方式執(zhí)行shell命令
[root@localhost python]# vim 1.py#!/bin/env python3import subprocessb=subprocess.run('ls -l /ltp | head -2', shell=True) # 執(zhí)行run方法,并將返回值賦值給b# total 184980# -rw-r--r--. 1 root root 10865 May 8 16:21 123.txtprint(b) # 執(zhí)行該run函數(shù)后返回的CompletedProcess類# CompletedProcess(args=’ls -l /ltp | head -2’, returncode=0)print(b.args) # 打印出CompletedProcess類的agrs屬性值,就是執(zhí)行的shell命令# ls -l /ltp | head -2print(b.returncode) # 打印命令執(zhí)行的狀態(tài)碼# 0
結(jié)果展示
[root@localhost python]# ./1.pytotal 184980-rw-r--r--. 1 root root 10865 May 8 16:21 123.txt
CompletedProcess(args=’ls -l /ltp | head -2’, returncode=0)2
ls -l /ltp | head -2
0
2.使用列表方式執(zhí)行
個(gè)人感覺方法二不好用,尤其是想要使用管道符號是,很難用
#!/bin/env python3import subprocessb = subprocess.run(['ls', '-l', '/ltp'])print(b)print(b.args)print(b.returncode)
執(zhí)行結(jié)果
[root@localhost python]# ./2.pytotal 10865-rw-r--r--. 1 root root 10865 May 8 16:21 123.txtCompletedProcess(args=[’ls’, ’-l’, ’/ltp’], returncode=0)[’ls’, ’-l’, ’/ltp’]0
3.捕獲腳本輸出
如果需要采集命令執(zhí)行的結(jié)果,可以傳入?yún)?shù)stdout=subprocess.PIPE[root@localhost python]# cat 3.py#!/bin/env python3import subprocess# 傳入stdout=subprocess.PIPE參數(shù)即可b=subprocess.run('ls -l /ltp | head -2', shell=True, stdout=subprocess.PIPE)print(b.stdout)
結(jié)果顯示
[root@localhost python]# ./1.pyb’total 184980n-rw-r--r--. 1 root root 10865 May 8 16:21 123.txtn’
4.檢測異常
示例1:模擬renturncode值不為0
傳入?yún)?shù)check=True,當(dāng)返回值不為0時(shí),就會(huì)拋出異常
[root@localhost python]# cat 1.py#!/bin/env python3import subprocessb=subprocess.run('ls -l /123 | head -2 && exit 1', shell=True, stdout=subprocess.PIPE, check=True)print(b.returncode)
執(zhí)行結(jié)果:返回了CalledProcessError 類型報(bào)錯(cuò)
[root@localhost python]# ./1.pyls: cannot access /123: No such file or directoryTraceback (most recent call last): File './1.py', line 3, in <module> b=subprocess.run('ls -l /123 | head -2 && exit 1', shell=True, stdout=subprocess.PIPE, check=True) File '/usr/local/python3/lib/python3.7/subprocess.py', line 487, in run output=stdout, stderr=stderr)subprocess.CalledProcessError: Command ’ls -l /123 | head -2 && exit 1’ returned non-zero exit status 1.# 返回了 CalledProcessError 類型報(bào)錯(cuò)
示例2:模擬執(zhí)行超時(shí)
返回 TimeoutExpired 異常
[root@localhost python]# vim 1.py#!/bin/env python3import subprocessb=subprocess.run('while 2>1;do sleep 1;done',timeout=3, shell=True, stdout=subprocess.PIPE, check=True)print(b.returncode)
顯示結(jié)果
[root@localhost python]# ./1.pyTraceback (most recent call last): File '/usr/local/python3/lib/python3.7/subprocess.py', line 474, in run stdout, stderr = process.communicate(input, timeout=timeout) File '/usr/local/python3/lib/python3.7/subprocess.py', line 939, in communicate stdout, stderr = self._communicate(input, endtime, timeout) File '/usr/local/python3/lib/python3.7/subprocess.py', line 1682, in _communicate self._check_timeout(endtime, orig_timeout) File '/usr/local/python3/lib/python3.7/subprocess.py', line 982, in _check_timeout raise TimeoutExpired(self.args, orig_timeout)subprocess.TimeoutExpired: Command ’while 2>1;do sleep 1;done’ timed out after 3 seconds
During handling of the above exception, another exception occurred:
Traceback (most recent call last): File './1.py', line 3, in <module> b=subprocess.run('while 2>1;do sleep 1;done',timeout=3, shell=True, stdout=subprocess.PIPE, check=True) File '/usr/local/python3/lib/python3.7/subprocess.py', line 479, in run stderr=stderr)subprocess.TimeoutExpired: Command ’while 2>1;do sleep 1;done’ timed out after 3 seconds
2、Popen類1.初步認(rèn)識Popen類
首先來看一下Popen類的構(gòu)造函數(shù)
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):
參數(shù)字符串或列表bufsize0 : 無緩沖
參數(shù) 字符串或列表 bufsize 0 : 無緩沖 1 : 行緩沖 其他正數(shù)值 :緩沖區(qū)大小 負(fù)數(shù)值 :采用默認(rèn)的系統(tǒng)緩沖(一般是全緩沖) executable 一般不用,args 字符串或列表 的第一項(xiàng)表示程序名 stdin stdout stderr None : 沒有任何重定向 繼承父進(jìn)程 PIPE : 創(chuàng)建管道 文件對象 文件描述符(整數(shù)) stderr也可以設(shè)置為stdout preexec_fn 鉤子函數(shù),在fork和exec之間執(zhí)行 close_fds unix 下執(zhí)行新進(jìn)程前是否關(guān)閉0/1/2之外的文件 windows 下不繼承還是繼承父進(jìn)程的文件描述 shell 若為True的話 : 在unix 下相當(dāng)于在args前面添加了 “/bin/bash” “-c' 在windows下,相當(dāng)于添加了'cmd.exe /c” cwd 設(shè)置工作目錄 env 設(shè)置環(huán)境變量 unviersal_newlines 各種換行符統(tǒng)一處理成'n' startupinfo windows下傳遞給CreateProcess的結(jié)構(gòu)體 creationflags windows下,傳遞CREATE_NEW_CONSOLE創(chuàng)建自己的控制臺窗口2.Popen的使用方法
1、subprocess.Popen([“cat”, “abc.txt”])2、subprocess.Popen(“cat abc.txt”, shell=True)
上面的第二種其實(shí)就相當(dāng)于:subprocess.Popen(['/bin/bash', “-c”, “cat abc.txt”])
示例:
[root@localhost python]# cat 3.py#!/bin/env python3import subprocessobj = subprocess.Popen('ls -l /ltp', shell=True, stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE, universal_newlines=True)error_info = obj.stderr.read()out_info = obj.stdout.read()result = out_info + error_infoprint(result)[root@localhost python]# ./3.pytotal 184980-rw-r--r--. 1 root root 10865 May 8 16:21 123.txt-rw-r--r--. 1 root root 802 Apr 21 09:42 ab.shdrwxr-xr-x. 3 root root 101 Apr 1 18:34 auth-rw-r--r--. 1 root root 5242880 Mar 18 13:20 bigfile-rwxrwxrwx. 1 root root 1392 Feb 5 09:24 dingding.sh
Popen類的對象方法
名稱 功能 poll() 檢查是否結(jié)束,設(shè)置返回值 wait() 等待結(jié)束,設(shè)置返回值 communicate() 參數(shù)是標(biāo)準(zhǔn)輸入,返回標(biāo)準(zhǔn)輸出和標(biāo)準(zhǔn)出錯(cuò) send_signal() 發(fā)動(dòng)信號(主要指linux下有用) terminate() 終止進(jìn)程,unix對應(yīng)的SIGTERM信號,windows下調(diào)用api函數(shù)TerminateProcess() kill() 殺死進(jìn)程(unix對應(yīng)SIGKILL信號),windows下同上 stdin stdout stderr 參數(shù)中指定PIPE時(shí)有用 pid 進(jìn)程id returncode 進(jìn)程返回值補(bǔ)充:其他方法
1、subeprocess.call(*args,**kwargs): call()方法調(diào)用Popen()執(zhí)行程序,并且等待它執(zhí)行完成2、subpeocess.check_call(*args, **kwargs): 調(diào)用上面的call(),如果返回值非零,返回異常3、subprocess.check_output(*args, **kwargs) : 調(diào)用Popen()執(zhí)行程序,并返回標(biāo)準(zhǔn)輸出
二、補(bǔ)充os模塊執(zhí)行外部命令1、os.system()方法示例:
[root@localhost python]# cat 4.py#!/bin/env python3import os# 變量ret接收命令執(zhí)行后的返回值ret = os.system(’ls -l /ltp |head -2’)print('n執(zhí)行成功' if ret == 0 else 'n執(zhí)行失敗')
執(zhí)行結(jié)果
[root@localhost python]# ./4.pytotal 184980-rw-r--r--. 1 root root 10865 May 8 16:21 123.txt
執(zhí)行成功
2、os.popen()用法與subprocess.Popen()類似,就不寫了
補(bǔ)充:subprocess.run()和subprocess.Popen()的執(zhí)行結(jié)果是寫入到緩存的,可以執(zhí)行結(jié)束后打印結(jié)果,不會(huì)實(shí)時(shí)在終端輸出;而os.system()是實(shí)時(shí)輸出到終端界面的;
以上就是Python執(zhí)行外部命令subprocess的詳細(xì)內(nèi)容,更多關(guān)于Python執(zhí)行外部命令的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. python中scrapy處理項(xiàng)目數(shù)據(jù)的實(shí)例分析2. Python中讀取文件名中的數(shù)字的實(shí)例詳解3. 在idea中為注釋標(biāo)記作者日期操作4. 通過Ajax方式綁定select選項(xiàng)數(shù)據(jù)的實(shí)例5. JSP頁面的靜態(tài)包含和動(dòng)態(tài)包含使用方法6. ASP.Net Core對USB攝像頭進(jìn)行截圖7. ASP.NET MVC使用Boostrap實(shí)現(xiàn)產(chǎn)品展示、查詢、排序、分頁8. .net如何優(yōu)雅的使用EFCore實(shí)例詳解9. 使用AJAX(包含正則表達(dá)式)驗(yàn)證用戶登錄的步驟10. ajax動(dòng)態(tài)加載json數(shù)據(jù)并詳細(xì)解析
