ash和Python是大多數自動化工程師最喜歡的編程語言。兩者都有利有弊,有時很難選擇,那么你應該使用哪一種。答案是:它取決于任務,范圍,背景和復雜性。
讓我們比較這兩種語言,以便更好地了解每種語言的選擇。
help(os.system)
#os.system(command):該方法在調用完shell腳本后,返回一個16位的二進制數,
#低位為殺死所調用腳本的信號號碼,高位為腳本的退出狀態碼,
#即腳本中exit 1的代碼執行后,os.system函數返回值的高位數則是1,如果低位數是0的情況下,
#則函數的返回值是0x0100,換算為十進制得到256。
#要獲得os.system的正確返回值,可以使用位移運算(將返回值右移8位)還原返回值:
>>> import os
>>> os.system("./test.sh")
hello python!
hello world!
256
>>> n>>8
1
help(os.system)
#os.popen(command):這種調用方式是通過管道的方式來實現,函數返回一個file對象,
#里面的內容是腳本輸出的內容(可簡單理解為echo輸出的內容),使用os.popen調用test.sh的情況
>> import os
>>> os.popen("./test.sh")
<open file './test.sh', mode 'r' at 0x7f6cbbbee4b0>
>>> f=os.popen("./test.sh")
>>> f
<open file './test.sh', mode 'r' at 0x7f6cbbbee540>
>>> f.readlines()
['hello python!\n', 'hello world!\n']
(1)commands.getstatusoutput(cmd),其以字符串的形式返回的是輸出結果和狀態碼,即(status,output)。
(2)commands.getoutput(cmd),返回cmd的輸出結果。
(3)commands.getstatus(file),返回ls -l file的執行結果字符串,調用了getoutput,不建議使用此方法
subprocess模塊,允許創建很多子進程,創建的時候能指定子進程和子進程的輸入、輸出、錯誤輸出管道,執行后能獲取輸出結果和執行狀態。
(1)subprocess.run():python3.5中新增的函數, 執行指定的命令, 等待命令執行完成后返回一個包含執行結果的CompletedProcess類的實例。
(2)subprocess.call():執行指定的命令, 返回命令執行狀態, 功能類似os.system(cmd)。
(3)subprocess.check_call():python2.5中新增的函數, 執行指定的命令, 如果執行成功則返回狀態碼, 否則拋出異常。
說明:subprocess.run(args, *, stdin=None, input=None, stdout=None, stderr=None, shell=False, timeout=None, check=False, universal_newlines=False)
subprocess.call(args, *, stdin=None, stdout=None, stderr=None, shell=False, timeout=None)
subprocess.check_call(args, *, stdin=None, stdout=None, stderr=None, shell=False, timeout=None)
args:表示shell指令,若以字符串形式給出shell指令,如"ls -l “則需要使shell=Ture。否則默認已數組形式表示shell變量,如"ls”,"-l"。
當使用比較復雜的shell語句時,可以先使用shlex模塊的shlex.split()方法來幫助格式化命令,然后在傳遞給run()方法或Popen。
# Stubs for subprocess
# Based on http://docs.python.org/2/library/subprocess.html and Python 3 stub
from typing import Sequence, Any, Mapping, Callable, Tuple, IO, Union, Optional, List, Text
_FILE=Union[None, int, IO[Any]]
_TXT=Union[bytes, Text]
_CMD=Union[_TXT, Sequence[_TXT]]
_ENV=Union[Mapping[bytes, _TXT], Mapping[Text, _TXT]]
# Same args as Popen.__init__
def call(args: _CMD,
bufsize: int=...,
executable: _TXT=...,
stdin: _FILE=...,
stdout: _FILE=...,
stderr: _FILE=...,
preexec_fn: Callable[[], Any]=...,
close_fds: bool=...,
shell: bool=...,
cwd: _TXT=...,
env: _ENV=...,
universal_newlines: bool=...,
startupinfo: Any=...,
creationflags: int=...) -> int: ...
def check_call(args: _CMD,
bufsize: int=...,
executable: _TXT=...,
stdin: _FILE=...,
stdout: _FILE=...,
stderr: _FILE=...,
preexec_fn: Callable[[], Any]=...,
close_fds: bool=...,
shell: bool=...,
cwd: _TXT=...,
env: _ENV=...,
universal_newlines: bool=...,
startupinfo: Any=...,
creationflags: int=...) -> int: ...
# Same args as Popen.__init__ except for stdout
def check_output(args: _CMD,
bufsize: int=...,
executable: _TXT=...,
stdin: _FILE=...,
stderr: _FILE=...,
preexec_fn: Callable[[], Any]=...,
close_fds: bool=...,
shell: bool=...,
cwd: _TXT=...,
env: _ENV=...,
universal_newlines: bool=...,
startupinfo: Any=...,
creationflags: int=...) -> bytes: ...
PIPE=... # type: int
STDOUT=... # type: int
class CalledProcessError(Exception):
returncode=0
# morally: _CMD
cmd=... # type: Any
# morally: Optional[bytes]
output=... # type: Any
def __init__(self,
returncode: int,
cmd: _CMD,
output: Optional[bytes]=...) -> None: ...
class Popen:
stdin=... # type: Optional[IO[Any]]
stdout=... # type: Optional[IO[Any]]
stderr=... # type: Optional[IO[Any]]
pid=0
returncode=0
def __init__(self,
args: _CMD,
bufsize: int=...,
executable: Optional[_TXT]=...,
stdin: Optional[_FILE]=...,
stdout: Optional[_FILE]=...,
stderr: Optional[_FILE]=...,
preexec_fn: Optional[Callable[[], Any]]=...,
close_fds: bool=...,
shell: bool=...,
cwd: Optional[_TXT]=...,
env: Optional[_ENV]=...,
universal_newlines: bool=...,
startupinfo: Optional[Any]=...,
creationflags: int=...) -> None: ...
def poll(self) -> int: ...
def wait(self) -> int: ...
# morally: -> Tuple[Optional[bytes], Optional[bytes]]
def communicate(self, input: Optional[_TXT]=...) -> Tuple[Any, Any]: ...
def send_signal(self, signal: int) -> None: ...
def terminate(self) -> None: ...
def kill(self) -> None: ...
def __enter__(self) -> 'Popen': ...
def __exit__(self, type, value, traceback) -> bool: ...
# Windows-only: STARTUPINFO etc.
STD_INPUT_HANDLE=... # type: Any
STD_OUTPUT_HANDLE=... # type: Any
STD_ERROR_HANDLE=... # type: Any
SW_HIDE=... # type: Any
STARTF_USESTDHANDLES=... # type: Any
STARTF_USESHOWWINDOW=... # type: Any
CREATE_NEW_CONSOLE=... # type: Any
CREATE_NEW_PROCESS_GROUP=... # type: Any
https://www.jb51.net/article/186301.htm