subprocess 模塊

来源:http://www.cnblogs.com/bigberg/archive/2017/07/08/7136952.html
-Advertisement-
Play Games

一、簡介 subprocess最早在2.4版本引入。用來生成子進程,並可以通過管道連接他們的輸入/輸出/錯誤,以及獲得他們的返回值。 subprocess用來替換多個舊模塊和函數: os.system os.spawn* os.popen* popen2.* commands.* 運行python的 ...


一、簡介

  subprocess最早在2.4版本引入。用來生成子進程,並可以通過管道連接他們的輸入/輸出/錯誤,以及獲得他們的返回值。

  subprocess用來替換多個舊模塊和函數:

  • os.system
  • os.spawn*
  • os.popen*
  • popen2.*
  • commands.*

  運行python的時候,我們都是在創建並運行一個進程,linux中一個進程可以fork一個子進程,並讓這個子進程exec另外一個程式。在python中,我們通過標準庫中的subprocess包來fork一個子進程,並且運行一個外部的程式。subprocess包中定義有數個創建子進程的函數,這些函數分別以不同的方式創建子進程,所欲我們可以根據需要來從中選取一個使用。另外subprocess還提供了一些管理標準流(standard stream)和管道(pipe)的工具,從而在進程間使用文本通信。   

二、舊有模塊的使用

1.os.system()

執行操作系統的命令,將結果輸出到屏幕,只返回命令執行狀態(0:成功,非 0 : 失敗)

import os

>>> a = os.system("df -Th")
Filesystem     Type   Size  Used Avail Use% Mounted on
/dev/sda3      ext4   1.8T  436G  1.3T  26% /
tmpfs          tmpfs   16G     0   16G   0% /dev/shm
/dev/sda1      ext4   190M  118M   63M  66% /boot

>>> a
0         # 0 表示執行成功


# 執行錯誤的命令
>>> res = os.system("list")
sh: list: command not found
>>> res
32512       # 返回非 0 表示執行錯誤

  

2. os.popen()

執行操作系統的命令,會將結果保存在記憶體當中,可以用read()方法讀取出來

import os

>>> res = os.popen("ls -l")

# 將結果保存到記憶體中
>>> print res
<open file 'ls -l', mode 'r' at 0x7f02d249c390>

# 用read()讀取內容
>>> print res.read()
total 267508
-rw-r--r--  1 root root    260968 Jan 27  2016 AliIM.exe
-rw-------. 1 root root      1047 May 23  2016 anaconda-ks.cfg
-rw-r--r--  1 root root   9130958 Nov 18  2015 apache-tomcat-8.0.28.tar.gz
-rw-r--r--  1 root root         0 Oct 31  2016 badblocks.log
drwxr-xr-x  5 root root      4096 Jul 27  2016 certs-build
drwxr-xr-x  2 root root      4096 Jul  5 16:54 Desktop
-rw-r--r--  1 root root      2462 Apr 20 11:50 Face_24px.ico

  

 三、subprocess模塊

1、subprocess.run()

>>> import subprocess
# python 解析則傳入命令的每個參數的列表
>>> subprocess.run(["df","-h"])
Filesystem            Size  Used Avail Use% Mounted on
/dev/mapper/VolGroup-LogVol00
                      289G   70G  204G  26% /
tmpfs                  64G     0   64G   0% /dev/shm
/dev/sda1             283M   27M  241M  11% /boot
CompletedProcess(args=['df', '-h'], returncode=0)

# 需要交給Linux shell自己解析,則:傳入命令字元串,shell=True
>>> subprocess.run("df -h|grep /dev/sda1",shell=True)
/dev/sda1             283M   27M  241M  11% /boot
CompletedProcess(args='df -h|grep /dev/sda1', returncode=0)

  

2、subprocess.call()

執行命令,返回命令的結果和執行狀態,0或者非0

>>> res = subprocess.call(["ls","-l"])
總用量 28
-rw-r--r--  1 root root     0 6月  16 10:28 1
drwxr-xr-x  2 root root  4096 6月  22 17:48 _1748
-rw-------. 1 root root  1264 4月  28 20:51 anaconda-ks.cfg
drwxr-xr-x  2 root root  4096 5月  25 14:45 monitor
-rw-r--r--  1 root root 13160 5月   9 13:36 npm-debug.log

# 命令執行狀態
>>> res
0

  

3、subprocess.check_call()

執行命令,返回結果和狀態,正常為0 ,執行錯誤則拋出異常

>>> subprocess.check_call(["ls","-l"])
總用量 28
-rw-r--r--  1 root root     0 6月  16 10:28 1
drwxr-xr-x  2 root root  4096 6月  22 17:48 _1748
-rw-------. 1 root root  1264 4月  28 20:51 anaconda-ks.cfg
drwxr-xr-x  2 root root  4096 5月  25 14:45 monitor
-rw-r--r--  1 root root 13160 5月   9 13:36 npm-debug.log
0

>>> subprocess.check_call(["lm","-l"])
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/usr/lib64/python2.7/subprocess.py", line 537, in check_call
    retcode = call(*popenargs, **kwargs)
  File "/usr/lib64/python2.7/subprocess.py", line 524, in call
    return Popen(*popenargs, **kwargs).wait()
  File "/usr/lib64/python2.7/subprocess.py", line 711, in __init__
    errread, errwrite)
  File "/usr/lib64/python2.7/subprocess.py", line 1327, in _execute_child
    raise child_exception
OSError: [Errno 2] No such file or directory

  

4、subprocess.getstatusoutput()

接受字元串形式的命令,返回 一個元組形式的結果,第一個元素是命令執行狀態,第二個為執行結果

#執行正確
>>> subprocess.getstatusoutput('pwd')
(0, '/root')

#執行錯誤
>>> subprocess.getstatusoutput('pd')
(127, '/bin/sh: pd: command not found')

  

5、subprocess.getoutput()

接受字元串形式的命令,放回執行結果

>>> subprocess.getoutput('pwd')
'/root'

  

6、subprocess.check_output()

執行命令,返回執行的結果,而不是列印

>>> res = subprocess.check_output("pwd")
>>> res
b'/root\n' # 結果以位元組形式返回

  

四、subprocess.Popen()

其實以上subprocess使用的方法,都是對subprocess.Popen的封裝,下麵我們就來看看這個Popen方法。

1、stdout

標準輸出

 

>>> res = subprocess.Popen("ls /tmp/yum.log", shell=True, stdout=subprocess.PIPE)  # 使用管道
>>> res.stdout.read()    # 標準輸出
b'/tmp/yum.log\n'

res.stdout.close()   # 關閉

 

  

2、stderr

標準錯誤

>>> import subprocess
>>> res = subprocess.Popen("lm -l",shell=True,stdout=subprocess.PIPE,stderr=subprocess.PIPE)
# 標準輸出為空
>>> res.stdout.read()
b''

#標準錯誤中有錯誤信息
>>> res.stderr.read()
b'/bin/sh: lm: command not found\n'

  

註意:上面的提到的標準輸出都為啥都需要等於subprocess.PIPE,這個又是啥呢?原來這個是一個管道,這個需要畫一個圖來解釋一下:

4、poll()

定時檢查命令有沒有執行完畢,執行完畢後返回執行結果的狀態,沒有執行完畢返回None

>>> res = subprocess.Popen("sleep 10;echo 'hello'",shell=True,stdout=subprocess.PIPE,stderr=subprocess.PIPE)
>>> print(res.poll())
None
>>> print(res.poll())
None
>>> print(res.poll())
0

  

5、wait()

等待命令執行完成,並且返回結果狀態

>>> obj = subprocess.Popen("sleep 10;echo 'hello'",shell=True,stdout=subprocess.PIPE,stderr=subprocess.PIPE)
>>> obj.wait()


# 中間會一直等待


0

  

6、terminate()

結束進程

import subprocess

>>> res = subprocess.Popen("sleep 20;echo 'hello'",shell=True,stdout=subprocess.PIPE,stderr=subprocess.PIPE)
>>> res.terminate()  # 結束進程
>>> res.stdout.read() 
b''

  

7、pid

獲取當前執行子shell的程式的進程號

import subprocess

>>> res = subprocess.Popen("sleep 5;echo 'hello'",shell=True,stdout=subprocess.PIPE,stderr=subprocess.PIPE)
>>> res.pid  # 獲取這個linux shell 的 進程號
2778

  

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • google已經將kotlin作為android開發的首選語言,然而我並不是android開發者,也不是java開發者,那麼我為什麼要學kotlin呢? 也許是心血來潮,也許是因為JB家出的編程語言必定會火,也許我只是JB家的忠實粉絲而已,不管怎麼樣吧,在此立一個flag,開啟我的kotlin學習之 ...
  • 一 概述 1.EL Expression Language,表達式語言,一種不同於編程語言的語言,用於訪問對象或者為對象賦值,取代JSP頁面中嵌套的java代碼,使頁面風格統一。 2.語法格式 expression既可以是屬性,也可以是字面值,還可以是算術表達式、關係表達式、邏輯表達式、條件表達式等 ...
  • 今天第一次接觸java的網頁編程,因為我之前是學習php的。所以學習jsp的話感覺是很相似的。第一次入門的程式是混編的形式,和php的用法相識,java規定的嵌入語言是用<% %>來表示是java的代碼,而php的話是使用<?php ?>來表示php的代碼塊。 這次的學習的效果如下: 代碼如下: 在 ...
  • 題目背景 很多學校流行一種比較的習慣。老師們很喜歡詢問,從某某到某某當中,分數最高的是多少。這讓很多學生很反感。 題目描述 不管你喜不喜歡,現在需要你做的是,就是按照老師的要求,寫一個程式,模擬老師的詢問。當然,老師有時候需要更新某位同學的成績 輸入輸出格式 輸入格式: 第一行,有兩個正整數 N 和 ...
  • 動態代理案例1:/*要求:運用Proxy動態代理來增強方法題目: 1.定義介面Fruit,其中有addFruit方法 2.定義實現類FruitImpl,實現Fruit介面 3.定義測試類,利用動態代理類的方式,增強addFruit方法*/ 1 import java.lang.reflect.Pro... ...
  • 文件內容如下: 現在看如何處理並轉成列表! 輸出結果如下: ...
  • K-th Number Time Limit: 20000MS Memory Limit: 65536K Total Submissions: 57427 Accepted: 19856 Case Time Limit: 2000MS Description You are working for ...
  • JSP的本質是Servlet源程式 每個JSP頁面在第一次被訪問時,JSP引擎將它翻譯成一個Servlet源程式,接著再把這個Serlvet源程式編譯成Servlet的class類文件。 然後再由Web容器(Servlet引擎)像調用普通Servlet程式一樣的方式來裝載和解釋執行這個Servlet ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...