python?subprocess.run()、subprocess.Popen()、subprocess.check_output()
Python的subprocess模塊是用于創(chuàng)建和管理子進(jìn)程的模塊。它提供了一種在Python中調(diào)用外部命令的方式,可以執(zhí)行系統(tǒng)命令、啟動(dòng)新的進(jìn)程、連接到子進(jìn)程的輸入/輸出管道等。
基本函數(shù)
subprocess.run
subprocess.run(args, *, stdin=None, input=None, stdout=None, stderr=None, capture_output=False, text=None, check=False, timeout=None, encoding=None, errors=None)
- 運(yùn)行指定的命令,并等待其完成。args參數(shù)是一個(gè)字符串或列表,表示要執(zhí)行的命令和參數(shù)。
- stdin、stdout和stderr參數(shù)分別用于指定子進(jìn)程的標(biāo)準(zhǔn)輸入、標(biāo)準(zhǔn)輸出和標(biāo)準(zhǔn)錯(cuò)誤輸出的處理方式。
- capture_output參數(shù)用于指定是否捕獲子進(jìn)程的輸出。
- check參數(shù)用于指定是否檢查子進(jìn)程的返回值,如果返回值不為0,則會(huì)拋出CalledProcessError異常。
- timeout參數(shù)用于指定子進(jìn)程的超時(shí)時(shí)間。
- encoding和errors參數(shù)用于指定輸入/輸出的編碼方式和錯(cuò)誤處理方式。
示例(只可以捕獲標(biāo)準(zhǔn)輸出,標(biāo)準(zhǔn)錯(cuò)誤不知道咋獲取,異常也沒拋。。。以后再看,現(xiàn)在有事)
import subprocess def execute_update_ip_sh(ip): """ 執(zhí)行修改配置文件 ip 腳本,將會(huì)修改ky_ai_solution.json、nginx.conf、vsftpd.conf 等文件中的ip """ try: # 構(gòu)建命令 command = ['/ky/update_ip.sh', ip] # 執(zhí)行命令并獲取輸出(這個(gè)出錯(cuò)直接就拋異常了,不能捕獲標(biāo)準(zhǔn)輸出和標(biāo)準(zhǔn)錯(cuò)誤) # result = subprocess.check_output(command, universal_newlines=True) result = subprocess.run(command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, universal_newlines=True) # 打印腳本的輸出 print(f"[{command}] 標(biāo)準(zhǔn)輸出:", result.stdout) # 打印腳本的錯(cuò)誤輸出(打印不出來(lái)) print(f"[{command}] 標(biāo)準(zhǔn)錯(cuò)誤輸出:", result.stderr) # 執(zhí)行成功,返回True return result.returncode == 0 # except subprocess.CalledProcessError as e: except Exception as e: # 執(zhí)行失敗,返回False print(e) print("ddddddddddddddddddddddddddddddddddddddd") return False if __name__ == "__main__": execute_update_ip_sh("192.168.1.140")
subprocess.Popen
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)
- 啟動(dòng)一個(gè)新的進(jìn)程,并返回一個(gè)Popen對(duì)象,可以用于與子進(jìn)程進(jìn)行交互。
- args參數(shù)和其他參數(shù)的含義與subprocess.run()函數(shù)相同。
subprocess.Popen參數(shù)解釋
- -
args
:要執(zhí)行的命令及其參數(shù)??梢允且粋€(gè)字符串或一個(gè)字符串列表。 - -
bufsize
:指定緩沖區(qū)的大小。默認(rèn)值為-1,表示使用系統(tǒng)默認(rèn)的緩沖區(qū)大小。 - -
executable
:指定要執(zhí)行的可執(zhí)行文件的路徑。默認(rèn)值為None,表示使用系統(tǒng)默認(rèn)的可執(zhí)行文件。 - -
stdin
:指定子進(jìn)程的標(biāo)準(zhǔn)輸入??梢允且粋€(gè)文件對(duì)象或一個(gè)文件描述符。默認(rèn)值為None,表示使用父進(jìn)程的標(biāo)準(zhǔn)輸入。 - -
stdout
:指定子進(jìn)程的標(biāo)準(zhǔn)輸出??梢允且粋€(gè)文件對(duì)象或一個(gè)文件描述符。默認(rèn)值為None,表示使用父進(jìn)程的標(biāo)準(zhǔn)輸出。 - -
stderr
:指定子進(jìn)程的標(biāo)準(zhǔn)錯(cuò)誤輸出??梢允且粋€(gè)文件對(duì)象或一個(gè)文件描述符。默認(rèn)值為None,表示使用父進(jìn)程的標(biāo)準(zhǔn)錯(cuò)誤輸出。 - -
preexec_fn
:在子進(jìn)程執(zhí)行前調(diào)用的可調(diào)用對(duì)象。默認(rèn)值為None,表示不調(diào)用任何函數(shù)。
示例:使用preexec_fn來(lái)將Popen對(duì)象綁定進(jìn)程組
當(dāng)使用shell=True
時(shí),subprocess.Popen()
函數(shù)創(chuàng)建的子進(jìn)程實(shí)際上是一個(gè)shell進(jìn)程,而不是直接執(zhí)行的命令。因此,向Popen
對(duì)象發(fā)送信號(hào)并不能直接影響到子進(jìn)程。
如果需要向通過shell執(zhí)行的命令發(fā)送信號(hào),可以使用os.killpg()
函數(shù)來(lái)發(fā)送信號(hào)給進(jìn)程組。具體步驟如下:
- 創(chuàng)建
Popen
對(duì)象時(shí),設(shè)置preexec_fn=os.setsid
,以創(chuàng)建一個(gè)新的進(jìn)程組。 - 使用
os.killpg()
函數(shù)發(fā)送信號(hào)給進(jìn)程組。
下面是一個(gè)示例,演示了如何向通過shell執(zhí)行的命令發(fā)送信號(hào):
import subprocess import os import signal # 通過shell執(zhí)行命令 process = subprocess.Popen('sleep 10', shell=True, preexec_fn=os.setsid) # 發(fā)送信號(hào)給進(jìn)程組 os.killpg(os.getpgid(process.pid), signal.SIGTERM)
在這個(gè)示例中,首先通過subprocess.Popen()
函數(shù)創(chuàng)建了一個(gè)通過shell執(zhí)行的命令。在創(chuàng)建Popen
對(duì)象時(shí),通過preexec_fn=os.setsid
設(shè)置了preexec_fn
參數(shù),以創(chuàng)建一個(gè)新的進(jìn)程組。
然后,使用os.killpg()
函數(shù)發(fā)送信號(hào)給進(jìn)程組。在這個(gè)示例中,發(fā)送了SIGTERM
信號(hào),即終止信號(hào)。
需要注意的是,向進(jìn)程組發(fā)送信號(hào)會(huì)影響到進(jìn)程組中的所有進(jìn)程。因此,如果有多個(gè)通過shell執(zhí)行的命令,它們都屬于同一個(gè)進(jìn)程組,發(fā)送信號(hào)時(shí)需要注意。
- -
close_fds
:指定是否在子進(jìn)程中關(guān)閉父進(jìn)程中打開的文件描述符。默認(rèn)值為True,表示關(guān)閉文件描述符。 - -
shell
:指定是否通過shell來(lái)執(zhí)行命令。默認(rèn)值為False,表示不通過shell執(zhí)行命令。
當(dāng)shell
參數(shù)設(shè)置為True
時(shí),表示通過shell來(lái)執(zhí)行命令。這意味著可以使用shell的語(yǔ)法和功能,比如管道、重定向、通配符等。在這種情況下,args
參數(shù)可以是一個(gè)字符串,表示要執(zhí)行的完整命令。
當(dāng)shell
參數(shù)設(shè)置為False
時(shí),表示不通過shell來(lái)執(zhí)行命令。這是默認(rèn)的行為。在這種情況下,args
參數(shù)應(yīng)該是一個(gè)字符串列表,其中第一個(gè)元素是要執(zhí)行的命令,后續(xù)元素是命令的參數(shù)。
需要注意的是,使用shell=True
時(shí),應(yīng)該謹(jǐn)慎處理輸入?yún)?shù),避免命令注入等安全問題。應(yīng)該始終對(duì)用戶輸入進(jìn)行驗(yàn)證和過濾,避免直接將用戶輸入拼接到命令中。
下面是一個(gè)示例,演示了shell
參數(shù)的用法:
import subprocess # 通過shell執(zhí)行命令 subprocess.Popen('ls -l', shell=True) # 不通過shell執(zhí)行命令 subprocess.Popen(['ls', '-l'])
在這個(gè)示例中,第一個(gè)Popen()
函數(shù)調(diào)用通過shell執(zhí)行了ls -l
命令,而第二個(gè)Popen()
函數(shù)調(diào)用不通過shell執(zhí)行了ls -l
命令。
使用shell執(zhí)行和不使用shell執(zhí)行的差異(以管道為例)
使用shell執(zhí)行:
下面是一個(gè)示例,演示了如何通過shell執(zhí)行命令并使用管道:
import subprocess # 通過shell執(zhí)行命令并使用管道 process = subprocess.Popen('ls -l | grep .txt', shell=True) # 等待子進(jìn)程結(jié)束 process.wait()
在這個(gè)示例中,通過subprocess.Popen()
函數(shù)的shell
參數(shù)設(shè)置為True
,表示通過shell執(zhí)行命令。然后,可以使用shell的管道語(yǔ)法|
將兩個(gè)命令連接起來(lái),實(shí)現(xiàn)輸出過濾。
不使用shell執(zhí)行:
import subprocess # 不通過shell執(zhí)行命令并使用管道 command1 = subprocess.Popen(['ls', '-l'], stdout=subprocess.PIPE) command2 = subprocess.Popen(['grep', '.txt'], stdin=command1.stdout, stdout=subprocess.PIPE) # 獲取命令2的輸出 output = command2.communicate()[0] # 打印輸出結(jié)果 print(output.decode())
在這個(gè)示例中,首先創(chuàng)建了兩個(gè)Popen()
對(duì)象,分別對(duì)應(yīng)兩個(gè)命令。command1
執(zhí)行了ls -l
命令,并將輸出通過管道傳遞給command2
。command2
執(zhí)行了grep .txt
命令,并將結(jié)果輸出到標(biāo)準(zhǔn)輸出。
通過command2.communicate()
方法獲取命令2的輸出,并將其打印出來(lái)。
不通過shell執(zhí)行命令時(shí),需要手動(dòng)設(shè)置命令之間的輸入輸出關(guān)系,以及處理命令的輸出。這需要更多的代碼來(lái)實(shí)現(xiàn),但也提供了更多的靈活性和控制能力。
-
cwd
:指定子進(jìn)程的當(dāng)前工作目錄。默認(rèn)值為None,表示使用父進(jìn)程的當(dāng)前工作目錄。-
env
:指定子進(jìn)程的環(huán)境變量??梢允且粋€(gè)字典,表示環(huán)境變量的鍵值對(duì)。默認(rèn)值為None,表示使用父進(jìn)程的環(huán)境變量。-
universal_newlines
:指定是否將輸入/輸出流以文本模式打開。默認(rèn)值為False,表示以二進(jìn)制模式打開。-
startupinfo
:指定子進(jìn)程的啟動(dòng)信息??梢允且粋€(gè)subprocess.STARTUPINFO對(duì)象,用于設(shè)置子進(jìn)程的一些屬性。默認(rèn)值為None。-
creationflags
:指定子進(jìn)程的創(chuàng)建標(biāo)志。默認(rèn)值為0,表示使用默認(rèn)的創(chuàng)建標(biāo)志。-
restore_signals
:指定是否在子進(jìn)程中恢復(fù)信號(hào)處理程序。默認(rèn)值為True,表示恢復(fù)信號(hào)處理程序。-
start_new_session
:指定是否在新的會(huì)話中啟動(dòng)子進(jìn)程。默認(rèn)值為False,表示在當(dāng)前會(huì)話中啟動(dòng)子進(jìn)程。-
pass_fds
:指定要傳遞給子進(jìn)程的文件描述符列表。默認(rèn)值為(),表示不傳遞文件描述符。-
encoding
:指定輸入/輸出的編碼方式。默認(rèn)值為None,表示使用系統(tǒng)默認(rèn)的編碼方式。-
errors
:指定輸入/輸出的錯(cuò)誤處理方式。默認(rèn)值為None,表示使用系統(tǒng)默認(rèn)的錯(cuò)誤處理方式。
總結(jié)
subprocess.Popen()
函數(shù)會(huì)返回一個(gè)Popen對(duì)象,可以用于與子進(jìn)程進(jìn)行交互。Popen對(duì)象具有一些常用的方法和屬性,比如communicate()
、wait()
、poll()
、terminate()
、kill()
、stdout
、stderr
和returncode
等。
Popen對(duì)象的常用方法和屬性
- communicate(input=None, timeout=None):與子進(jìn)程進(jìn)行交互,發(fā)送輸入并獲取輸出。
- wait():等待子進(jìn)程結(jié)束,并返回其返回值。
- poll():檢查子進(jìn)程是否結(jié)束,如果結(jié)束則返回其返回值,否則返回None。
- terminate():終止子進(jìn)程。
- kill():殺死子進(jìn)程。
- stdout:子進(jìn)程的標(biāo)準(zhǔn)輸出。
- stderr:子進(jìn)程的標(biāo)準(zhǔn)錯(cuò)誤輸出。
- returncode:子進(jìn)程的返回值。
下面是一個(gè)使用subprocess模塊的示例,演示了如何執(zhí)行一個(gè)系統(tǒng)命令并獲取其輸出:
import subprocess # 執(zhí)行系統(tǒng)命令 result = subprocess.run(['ls', '-l'], capture_output=True, text=True) # 獲取命令的輸出 output = result.stdout print(output)
以上代碼會(huì)執(zhí)行ls -l
命令,并將其輸出打印出來(lái)。
需要注意的是,subprocess模塊在執(zhí)行外部命令時(shí),存在一定的安全風(fēng)險(xiǎn),特別是在使用shell=True參數(shù)時(shí)。因此,在使用subprocess模塊時(shí),應(yīng)該謹(jǐn)慎處理輸入?yún)?shù),避免命令注入等安全問題。
subprocess.check_output
subprocess.check_output
函數(shù)是Python標(biāo)準(zhǔn)庫(kù)subprocess
模塊中的一個(gè)函數(shù),用于執(zhí)行系統(tǒng)命令并返回命令的輸出結(jié)果。
它的用法是:
subprocess.check_output(args, *, stdin=None, stderr=None, shell=False, universal_newlines=False)
其中,args
是一個(gè)字符串或者字符串列表,表示要執(zhí)行的命令及其參數(shù)。stdin
、stderr
、shell
和universal_newlines
是可選參數(shù),用于控制輸入、錯(cuò)誤輸出、是否使用shell以及輸出結(jié)果的編碼方式。
check_output
函數(shù)會(huì)執(zhí)行指定的命令,并等待命令執(zhí)行完成。如果命令執(zhí)行成功,它會(huì)返回命令的輸出結(jié)果(以字節(jié)字符串的形式),如果命令執(zhí)行失敗,它會(huì)拋出一個(gè)CalledProcessError
異常。
在上述代碼中,subprocess.check_output
函數(shù)被用于執(zhí)行一些系統(tǒng)命令,如ip addr show
和ip route show
,以獲取當(dāng)前以太網(wǎng)接口的IP配置信息。
到此這篇關(guān)于python subprocess.run()、subprocess.Popen()、subprocess.check_output()的文章就介紹到這了,更多相關(guān)python subprocess 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
用python 實(shí)現(xiàn)在不確定行數(shù)情況下多行輸入方法
今天小編就為大家分享一篇用python 實(shí)現(xiàn)在不確定行數(shù)情況下多行輸入方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來(lái)看看吧2019-01-01使用Django實(shí)現(xiàn)商城驗(yàn)證碼模塊的方法
本文主要涉及圖形驗(yàn)證碼的相關(guān)功能,主要包括,圖形驗(yàn)證碼獲取、驗(yàn)證碼文字存儲(chǔ)、驗(yàn)證碼生成等。需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-06-06跟老齊學(xué)Python之有點(diǎn)簡(jiǎn)約的元組
元組和列表十分類似,但是元組是不可變的.也就是說(shuō)你不能修改元組。元組通過圓括號(hào)中用逗號(hào)分割的項(xiàng)目定義。元組通常用在使語(yǔ)句或用戶定義的函數(shù)能夠安全地采用一組值的時(shí)候,即被使用的元組的值不會(huì)改變。2014-09-09520使用Python實(shí)現(xiàn)“我愛你”表白
這篇文章主要介紹了520使用Python實(shí)現(xiàn)“我愛你”表白,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05keras 解決加載lstm+crf模型出錯(cuò)的問題
這篇文章主要介紹了keras 解決加載lstm+crf模型出錯(cuò)的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來(lái)看看吧2020-06-06