1. 程式人生 > >關於os.popen你可能不知道的

關於os.popen你可能不知道的

前言

使用python執行一些命令或呼叫第三方工具是比較常見的情況,實現的方法也很多,如os.system()os.popen()subprocess模組中的函式等等,本文主要介紹一下os.popen(),可能隱藏著你不曾接觸的知識點,同時會結合例項進行說明,便於理解。

 

語法格式

首先我們看下os.popen的語法格式,如下:

os.popen(cmd, mode='r', buffering=-1)

 

引數說明:

Command --- 呼叫的命令;

mode --- 模式許可權可以是 'r'(預設) 'w'

bufsize -- 指明瞭檔案需要的緩衝大小:

0意味著無緩衝;1意味著行緩衝;其它正值表示使用引數大小的緩衝(大概值,以位元組為單位);負的bufsize意味著使用系統的預設值。

 

官方釋義:

Open a pipe to or from command cmd. The return value is an open file object connected to the pipe, which can be read or written depending on whether mode is 'r' (default) or 'w'.

從命令cmd開啟一個管道,返回值是連線管道的檔案物件,通過該物件可以進行讀或寫。

 

知識點梳理

 

1.  返回值是檔案物件

注意:返回值是檔案物件既然是檔案物件,使用完就應該關閉,對吧?!不信網上搜一下,一大把文章提到這個os.popen都是忘記關閉檔案物件的。 所以,推薦的寫法是:

  with os.popen(command, "r") as p:
    r = p.read()


     至於with的用法就不多講了,使用它,不需要顯式的寫p.close()

 

2.  非阻塞

通俗的講,非阻塞就是os.popen不會等cmd命令執行完畢就繼續下面的程式碼了,不信?!看下面程式碼例項:

1.png

從上面例項可知,os.popen執行開啟Uedit32.exe這個工具,但從實際執行結果看,Uedit32.exe還沒開啟,就直接進入了下一條語句,列印了“hello 小蟒社群”。在某些應用場景,可能這並不是你期望的行為,那如何讓命令執行完後,再執行下一句呢?

處理方法是使用read()readlines()對命令的執行結果進行讀操作。

 

3.  完全阻塞

上面寫了該函式是非阻塞的,現在怎麼又變成完全阻塞的呢?感覺一頭霧水了吧。本質上os.popen是非阻塞的,為了實現阻塞的效果,我們使用read()readlines()對命令結果進行讀,由此產生了阻塞的效果。但是,如果你的命令執行無法退出或進入互動模式,這種“讀”將形成完全阻塞的情況,表現的像程式卡住了。

看下面程式碼例項1

2.png

os.popen執行了ping  127.0.0.1  -t 該命令會一直執行,除非CTRL+C強制退出,因而,執行readlines讀取命令輸出時會造成卡住。

 

程式碼例項2

使用os.popen執行sqlplus命令對資料庫進行操作的場景,如果sqlplus執行失敗,會進入互動模式,如圖所示,此時使用readlines()讀取執行結果時也會卡死,效果如上圖:

3.png

 

總結

os.popen()在大多數場景都是挺好用方便的,但是也有坑!! 具體應用中,需要注意下。

建議:

1.       在需要讀取命令執行結果時,避免在命令無法退出或進入互動模式的場景應用os.popen;

2.       os.popen()無法滿足需求時,可以考慮subprocess.Popen();


其他資源

關於python學習、分享、交流,筆者開通了微信公眾號【小蟒社群】,感興趣的朋友可以關注下,歡迎加入,建立屬於我們自己的小圈子,一起學python