Python上下文管理器
本節(jié)嚴(yán)格意義上并非新的重定向方式,而是利用Pyhton上下文管理器優(yōu)化上節(jié)的代碼實(shí)現(xiàn)。借助于上下文管理器語法,可不必向重定向使用者暴露sys.stdout。
首先考慮輸出抑制,基于上下文管理器語法實(shí)現(xiàn)如下:
importsys,cStringIO,contextlib
classDummyFile:
defwrite(self,outStr):pass
@contextlib.contextmanager
defMuteStdout():
savedStdout=sys.stdout
sys.stdout=cStringIO.StringIO()#DummyFile()
try:
yield
exceptException:#捕獲到錯誤時,屏顯被抑制的輸出(該處理并非必需)
content,sys.stdout=sys.stdout,savedStdout
printcontent.getvalue()#;raise
#finally:
sys.stdout=savedStdout
使用示例如下:
withMuteStdout():
print"I'llshowupwhenisexecuted!"#不屏顯不寫入
raise#屏顯上句
print"I'mhidingmyselfsomewhere:)"#不屏顯
再考慮更通用的輸出重定向:
importos,sys
fromcontextlibimportcontextmanager
@contextmanager
defRedirectStdout(newStdout):
savedStdout,sys.stdout=sys.stdout,newStdout
try:
yield
finally:
sys.stdout=savedStdout
使用示例如下:
defGreeting():print'Hello,boss!'
withopen('out.txt',"w+")asfile:
print"I'mwritingtoyou..."#屏顯
withRedirectStdout(file):
print'Ihopethisletterfindsyouwell!'#寫入文件
print'Checkyourmailbox.'#屏顯
withopen(os.devnull,"w+")asfile,RedirectStdout(file):
Greeting()#不屏顯不寫入
print'Ideserveapayraise:)'#不屏顯不寫入
print'DidyouhearwhatIsaid?'#屏顯
可見,with內(nèi)嵌塊里的函數(shù)和print語句輸出均被重定向。注意,上述示例不是線程安全的,主要適用于單線程。
當(dāng)函數(shù)被頻繁調(diào)用時,建議使用裝飾器包裝該函數(shù)。這樣,僅需修改該函數(shù)定義,而無需在每次調(diào)用該函數(shù)時使用with語句包裹。示例如下:
importsys,cStringIO,functools
defMuteStdout(retCache=False):
defdecorator(func):
@functools.wraps(func)
defwrapper(*args,**kwargs):
savedStdout=sys.stdout
sys.stdout=cStringIO.StringIO()
try:
ret=func(*args,**kwargs)
ifretCache==True:
ret=sys.stdout.getvalue().strip()
finally:
sys.stdout=savedStdout
returnret
returnwrapper
returndecorator
若裝飾器MuteStdout的參數(shù)retCache為真,外部調(diào)用func()函數(shù)時將返回該函數(shù)內(nèi)部print輸出的內(nèi)容(可供屏顯);若retCache為假,外部調(diào)用func()函數(shù)時將返回該函數(shù)的返回值(抑制輸出)。
MuteStdout裝飾器使用示例如下:
@MuteStdout(True)
defExclaim():print'Iamproudofmyself!'
@MuteStdout()
defMumble():print'Ilackconfidence...';return'sad'
printExclaim(),Exclaim.__name__#屏顯'Iamproudofmyself!Exclaim'
printMumble(),Mumble.__name__#屏顯'sadMumble'
在所有線程中,被裝飾函數(shù)執(zhí)行期間,sys.stdout都會被MuteStdout裝飾器劫持。而且,函數(shù)一經(jīng)裝飾便無法移除裝飾。因此,使用該裝飾器時應(yīng)慎重考慮場景。
接著,考慮創(chuàng)建RedirectStdout裝飾器:
defRedirectStdout(newStdout=sys.stdout):
defdecorator(func):
defwrapper(*args,**kwargs):
savedStdout,sys.stdout=sys.stdout,newStdout
try:
returnfunc(*args,**kwargs)
finally:
sys.stdout=savedStdout
returnwrapper
returndecorator
使用示例如下:
file=open('out.txt',"w+")
@RedirectStdout(file)
defFunNoArg():print'Noargument.'
@RedirectStdout(file)
defFunOneArg(a):print'Oneargument:',a
defFunTwoArg(a,b):print'Twoarguments:%s,%s'%(a,b)
FunNoArg()#寫文件'Noargument.'
FunOneArg(1984)#寫文件'Oneargument:1984'
RedirectStdout()(FunTwoArg)(10,29)#屏顯'Twoarguments:10,29'
printFunNoArg.__name__#屏顯'wrapper'(應(yīng)顯示'FunNoArg')
file.close()
注意FunTwoArg()函數(shù)的定義和調(diào)用與其他函數(shù)的不同,這是兩種等效的語法。此外,RedirectStdout裝飾器的最內(nèi)層函數(shù)wrapper()未使用"functools.wraps(func)"修飾,會丟失被裝飾函數(shù)原有的特殊屬性(如函數(shù)名、文檔字符串等)。
以上內(nèi)容為大家介紹了Python上下文管理器,希望對大家有所幫助,如果想要了解更多Python相關(guān)知識,請關(guān)注IT培訓(xùn)機(jī)構(gòu):千鋒教育。

相關(guān)推薦HOT
更多>>
python cmd中怎么運(yùn)行python文件
運(yùn)行Python,可在交互模式下運(yùn)行,或者命令行中。命令行中運(yùn)行的是Python的.py文件。cmd中執(zhí)行Python程序的方法,如下打開cmdC:\Users\Administ...詳情>>
2023-11-06 14:37:04
pythondict是啥意思
Python中的dict表示的字典數(shù)據(jù)類型。字典是另一種可變?nèi)萜髂P停铱纱鎯θ我忸愋蛯ο蟆W值涫莗ython語言中唯一的映射類型映射類型對象里哈希值...詳情>>
2023-11-06 13:56:05
python如何創(chuàng)建模塊
Python模塊(Module),是一個Python文件,以.py結(jié)尾,包含了Python對象定義和Python語句。模塊讓你能夠有邏輯地組織你的Python代碼段。把相關(guān)的...詳情>>
2023-11-06 13:36:31
python解釋器寫代碼如何換行
使用以下方法可以使Python的代碼換行:用反斜杠鏈接多行代碼,示例輸出:55a=22+\33printa用小括號括起來,示例輸出:55a=(22+33)printa提示:針...詳情>>
2023-11-06 13:17:12熱門推薦
python division如何取整
沸pythonsort()報錯是什么原因
熱python學(xué)習(xí)python編程可以從事什么工作崗位
熱python基礎(chǔ)知識
新pythonint函數(shù)怎么用
python怎么打印輸出
python模塊下載后怎么安裝
python怎么傳遞不定參數(shù)
python怎么導(dǎo)包
python怎么注釋多行代碼
python如何安裝pip
python cmd中怎么運(yùn)行python文件
python標(biāo)準(zhǔn)庫是什么
pythondict是啥意思
技術(shù)干貨







快速通道 更多>>
-
課程介紹
點(diǎn)擊獲取大綱 -
就業(yè)前景
查看就業(yè)薪資 -
學(xué)習(xí)費(fèi)用
了解課程價格 -
優(yōu)惠活動
領(lǐng)取優(yōu)惠券 -
學(xué)習(xí)資源
領(lǐng)3000G教程 -
師資團(tuán)隊
了解師資團(tuán)隊 -
實(shí)戰(zhàn)項目
獲取項目源碼 -
開班地區(qū)
查看來校路線