Python下調(diào)用Linux的Shell命令的方法
有時候難免需要直接調(diào)用Shell命令來完成一些比較簡單的操作,比如mount一個文件系統(tǒng)之類的。那么我們使用Python如何調(diào)用Linux的Shell命令?下面來介紹幾種常用的方法:
1. os 模塊
1.1. os模塊的exec方法族
Python的exec系統(tǒng)方法同Unix的exec系統(tǒng)調(diào)用是一致的。這些方法適用于在子進(jìn)程中調(diào)用外部程序的情況,因為外部程序會替換當(dāng)前進(jìn)程的代碼,不會返回。( 這個看了點 help(os) --> search "exec" 的相關(guān)介紹,但是沒太搞明白咋使用)
1.2. os模塊的system方法
system方法會創(chuàng)建子進(jìn)程運(yùn)行外部程序,方法只返回外部程序的運(yùn)行結(jié)果。這個方法比較適用于外部程序沒有輸出結(jié)果的情況。
>>> import os
>>> os.system("echo \"Hello World\"") # 直接使用os.system調(diào)用一個echo命令
Hello World ——————> 打印命令結(jié)果
0 ——————> What's this ? 返回值?
>>> val = os.system("ls -al | grep \"log\" ") # 使用val接收返回值
-rw-r--r-- 1 root root 6030829 Dec 31 15:14 log ——————> 此時只打印了命令結(jié)果
>>> print val
0 ——————> 注意,此時命令正常運(yùn)行時,返回值是0
>>> val = os.system("ls -al | grep \"log1\" ")
>>> print val
256 ——————> 使用os.system調(diào)用一個沒有返回結(jié)果的命令,返回值為256~
>>>
注意:上面說了,此方法脂肪會外部程序的結(jié)果,也就是os.system的結(jié)果,所以如果你想接收命令的返回值,接著向下看~
1.3. os模塊的popen方法
當(dāng)需要得到外部程序的輸出結(jié)果時,本方法非常有用。比如使用urllib調(diào)用Web API時,需要對得到的數(shù)據(jù)進(jìn)行處理。os.popen(cmd) 要得到命令的輸出內(nèi)容,只需再調(diào)用下read()或readlines()等 如a=os.popen(cmd).read()
>>> os.popen('ls -lt') # 調(diào)用os.popen(cmd)并不能得到我們想要的結(jié)果
<open file 'ls -lt ', mode 'r' at 0xb7585ee8>
>>> print os.popen('ls -lt').read() # 調(diào)用read()方法可以得到命令的結(jié)果
total 6064
-rwxr-xr-x 1 long long 23 Jan 5 21:00 hello.sh
-rw-r--r-- 1 long long 147 Jan 5 20:26 Makefile
drwxr-xr-x 3 long long 4096 Jan 2 19:37 test
-rw-r--r-- 1 root root 6030829 Dec 31 15:14 log
drwxr-xr-x 2 long long 4096 Dec 28 09:36 pip_build_long
drwx------ 2 Debian-gdm Debian-gdm 4096 Dec 23 19:08 pulse-gylJ5EL24GU9
drwx------ 2 long long 4096 Jan 1 1970 orbit-long
>>> val = os.popen('ls -lt').read() # 使用變量可以接收命令返回值
>>> if "log" in val: # 我們可以使用in來判斷返回值中有木有一個字符串
... print "Haha,there is the log"
... else:
... print "No,not happy"
...
Haha,there is the log
2. commands 模塊
使用commands模塊的getoutput方法,這種方法同popend的區(qū)別在于popen返回的是一個文件句柄,而本方法將外部程序的輸出結(jié)果當(dāng)作字符串返回,很多情況下用起來要更方便些。
主要方法:
- commands.getstatusoutput(cmd) 返回(status, output)
- commands.getoutput(cmd) 只返回輸出結(jié)果
- commands.getstatus(file) 返回ls -ld file的執(zhí)行結(jié)果字符串,調(diào)用了getoutput,不建議使用此方法
long@zhouyl:/tmp/tests$ python
Python 2.7.3 (default, Jan 2 2013, 16:53:07)
[GCC 4.7.2] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> import commands
>>> commands.getstatusoutput('ls -lt') # 返回(status, output)
(0, 'total 5900\n-rwxr-xr-x 1 long long 23 Jan 5 21:34 hello.sh\n-rw-r--r-- 1 long long 147 Jan 5 21:34 Makefile\n-rw-r--r-- 1 long long 6030829 Jan 5 21:34 log')
>>> commands.getoutput('ls -lt') # 返回命令的輸出結(jié)果(貌似和Shell命令的輸出格式不同哈~)
'total 5900\n-rwxr-xr-x 1 long long 23 Jan 5 21:34 hello.sh\n-rw-r--r-- 1 long long 147 Jan 5 21:34 Makefile\n-rw-r--r-- 1 long long 6030829 Jan 5 21:34 log'
>>> commands.getstatus('log') # 調(diào)用commands.getoutput中的命令對'log'文件進(jìn)行相同的操作
'-rw-r--r-- 1 long long 6030829 Jan 5 21:34 log'
>>>
3. subprocess模塊
根據(jù)Python官方文檔說明,subprocess模塊用于取代上面這些模塊。有一個用Python實現(xiàn)的并行ssh工具—mssh,代碼很簡短,不過很有意思,它在線程中調(diào)用subprocess啟動子進(jìn)程來干活。
>>> from subprocess import call >>> call(["ls", "-l"])
subprocess與system相比的優(yōu)勢是它更靈活(你可以得到標(biāo)準(zhǔn)輸出,標(biāo)準(zhǔn)錯誤,“真正”的狀態(tài)代碼,更好的錯誤處理,等..)。我認(rèn)為使用os.system已過時,或即將過時。
4. 眾方法的比較以及總結(jié)
4.1. 關(guān)于 os.system
os.system("some_command with args")將命令以及參數(shù)傳遞給你的系統(tǒng)shell,這很好,因為你可以用這種方法同時運(yùn)行多個命令并且可以設(shè)置管道以及輸入輸出重定向。比如:
os.system("some_command < input_file | another_command > output_file")
然而,雖然這很方便,但是你需要手動處理shell字符的轉(zhuǎn)義,比如空格等。此外,這也只能讓你運(yùn)行簡單的shell命令而且不能運(yùn)行外部程序。
4.2. 關(guān)于os.popen
使用stream = os.popen("some_command with args")也能做與os.system一樣的事,與os.system不同的是os.popen會給你一個像文件的對象從而你可以使用它來訪問哪個程序的標(biāo)準(zhǔn)輸入、輸出。而且popen還有三個變種都是在I/O處理上有輕微不同。假如你通過一個字符串傳遞所有東西,你的命令會傳遞給shell;如果你通過一個列表傳遞他們,你不用擔(dān)心逃避任何事。
4.3. 關(guān)于subprocess.popen
subprocess模塊的Popen類,意圖作為os.popen的替代,但是因為其很全面所以比os.popen要顯得稍微復(fù)雜,使用起來需要學(xué)習(xí)哦~~。
比如你可以使用 print Popen("echo Hello World", stdout=PIPE, shell=True).stdout.read() 來替代 print os.popen("echo Hello World").read()。但是相比之下它使用一個統(tǒng)一的類包括4中不同的popen函數(shù)還是不錯的。
4.4. 關(guān)于subprocess.call
subprocess模塊的call函數(shù)。它基本上就像Popen類并都使用相同的參數(shù),但是它只簡單的等待命令完成并給你返回代碼。比
如:
return_code = subprocess.call("echo Hello World", shell=True)
os模塊中還有C中那樣的fork/exec/spawn函數(shù),但是我不建議直接使用它們。subprocess可能更加適合你。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- CentOS7中使用shell腳本安裝python3.8環(huán)境(推薦)
- 使用python執(zhí)行shell腳本 并動態(tài)傳參 及subprocess的使用詳解
- shell自動安裝python3的腳本寫法
- 在python shell中運(yùn)行python文件的實現(xiàn)
- python3模擬實現(xiàn)xshell遠(yuǎn)程執(zhí)行l(wèi)inux命令的方法
- 詳解python執(zhí)行shell腳本創(chuàng)建用戶及相關(guān)操作
- Python 運(yùn)行 shell 獲取輸出結(jié)果的實例
- 在python 中實現(xiàn)運(yùn)行多條shell命令
- Python調(diào)用shell cmd方法代碼示例解析
相關(guān)文章
Python 如何定義匿名或內(nèi)聯(lián)函數(shù)
這篇文章主要介紹了Python 如何定義匿名或內(nèi)聯(lián)函數(shù),文中講解非常細(xì)致,代碼幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下2020-08-08
Python socket實現(xiàn)的文件下載器功能示例
這篇文章主要介紹了Python socket實現(xiàn)的文件下載器功能,結(jié)合實例形式分析了Python使用socket模塊實現(xiàn)的文件下載器客戶端與服務(wù)器端相關(guān)操作技巧,需要的朋友可以參考下2019-11-11
詳解Django關(guān)于StreamingHttpResponse與FileResponse文件下載的最優(yōu)方法
這篇文章主要介紹了詳解Django關(guān)于StreamingHttpResponse與FileResponse文件下載的最優(yōu)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01
Python使用Matplotlib模塊時坐標(biāo)軸標(biāo)題中文及各種特殊符號顯示方法
這篇文章主要介紹了Python使用Matplotlib模塊時坐標(biāo)軸標(biāo)題中文及各種特殊符號顯示方法,結(jié)合具體實例分析了Python使用Matplotlib模塊過程中針對中文及特殊符號的顯示方法,需要的朋友可以參考下2018-05-05
pytorch 利用lstm做mnist手寫數(shù)字識別分類的實例
今天小編就為大家分享一篇pytorch 利用lstm做mnist手寫數(shù)字識別分類的實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-01-01

