前言 with語句的使用給我們帶來了很多的便利,最常用的可能就是關閉一個文件,釋放一把鎖。 既然with語句這麼好用,那我也想讓我自己寫的代碼也能夠使用with語句,該怎麼實現? 下麵具體介紹怎樣實現一個自己的with語句 使用類實現 要想使用with語句,那就要遵循with語句的使用規矩,也就是上 ...
前言
with語句的使用給我們帶來了很多的便利,最常用的可能就是關閉一個文件,釋放一把鎖。
既然with語句這麼好用,那我也想讓我自己寫的代碼也能夠使用with語句,該怎麼實現?
下麵具體介紹怎樣實現一個自己的with語句
使用類實現
要想使用with語句,那就要遵循with語句的使用規矩,也就是上下文管理器協議
這個協議提示我們要在類中去實現兩個特殊方法,enter(self)和exit(self, exc_type, exc_val, exc_tb)
有了這兩個方法,就可以使用with語句執行了,它的執行過程為先執行enter(self)方法,然後執行你添加在with下的代碼,然後最後執行exit方法
如果在執行的過程中出現了異常,那麼會自動執行exit方法,
class ErrorProduce:
def __init__(self):
pass
def produce(self):
raise RuntimeError('ERROR')
class MyWith:
def __enter__(self):
return ErrorProduce()
def __exit__(self, exc_type, exc_val, exc_tb):
if exc_type:
print("有錯")
#print(exc_val)
#return True
else:
print("完美執行")
with MyWith() as E:
E.produce()
將上述代碼運行,會先列印出“有錯”,然後再列印出相關錯誤,然後程式由於錯誤終止,也就是說,在產生了錯誤後,還是執行了exit中的代碼
如果將exit中的兩個註釋符號去掉的話,那麼再次運行程式不會報錯,但會將exc_val中的錯誤信息列印出來,這裡相當於通過exit方法將錯誤抹除掉了,抹除方式為返回True,但也把with中拋出錯誤後的所有代碼都跳過了。
使用函數實現
通過引入contextlib模塊中的contextmanager裝飾器,裝飾函數使函數可以使用with語句
from contextlib import contextmanager
class ErrorProduce:
def __init__(self):
pass
def produce(self):
raise RuntimeError('ERROR')
@contextmanager
def my_with():
try:
yield ErrorProduce()
except Exception as e:
print(e)
#raise
finally:
print('收尾')
with my_with() as E:
E.produce()
print('aa')
函數中使用了yield,那麼它也就變成了生成器,通過yield來劃分相當於類中的enter和exit,yield返回的值也就是可以用作as後的值
上段代碼中我們發現使用了try,except,finally異常處理的語句,這些語句是必要的嗎?
通過測試,如果沒有這些異常處理,那麼with語句還是可以執行的,這裡的意思是可以使用with語句,不會報不能使用with的錯誤,但是使用後,只有with下的語句中沒有拋出異常,那麼yield語句後邊的代碼才會執行,一旦有異常拋出,仍然是程式運行終止。所以,一般還是要配合try,except,finally語句來構造。
還有一點要註意,使用這種方式,異常被except捕捉後,那麼這個異常就失效了,一般的我們還是會需要在except中再次將捕捉的異常拋出,也就是在後面加上raise,因為我們使用with只是為了做好一個收尾,如關閉文件句柄。或者另一種方式,我們乾脆就不寫except語句,直接寫finally語句了。
- 文章中出現的棕色加粗的單詞,是由於markdown-here將特殊方法兩邊的下劃線轉換產生的,原意為特殊方法
-
Bootstrap -- 插件: 模態框、滾動監聽、標簽頁 1. 模態框(Modal): 覆蓋在父窗體上的子窗體。 使用模態框: <!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type" content="text/html; cha ...
-
作者按:《每天一個設計模式》旨在初步領會設計模式的精髓,目前採用 和`python`兩種語言實現。誠然,每種設計模式都有多種實現方式,但此小冊只記錄最直截了當的實現方式 :) 原文地址是: "《每天一個設計模式之組合模式》" 歡迎關註個人技術博客: "godbmw.com" 。每周 1 篇原創技術分 ...
-
使用橋接模式可以將類型的抽象和具體實現進行分離,兩者通過橋接模式進行關聯,從而達到解耦 介紹 橋接模式屬於結構型模式。在現實世界中,我們裝修房子時,佈線的工人和安裝空調的工人之間可以同時工作,不用互相依賴。而對於屋主人來講也不用關係他們具體時怎麼工作的,只需要等他們完成即可。在軟體開發中,當我們面對 ...
-
建造者模式 建造者模式適用場景: 建造一個複雜的對象適用,將構建對象的過程分開,每個類單獨構造對象的一部分,最後組裝起來,返回我們需要的對象。 下麵的例子主要講解構造一個飛船 Demo: //要獲得的對象,但是各個組件要拆分開,讓對應的類去實現 class AirShip { private Orb ...
-
見名知其意,適配器可用於對多個不相容介面提供適配橋梁 介紹 適配器模式屬於結構型模式。在現實世界中,這個模式適用的較為廣泛,比如 DIY 一些電子產品,主要元器件提供的是標準介面,那麼無論我們購買什麼品牌的元器件,最終都能組裝起來正常運行。 類圖描述 由上圖可知,我們通過定義 IAdvancedMe ...
-
面向對象設計原則 概述 對於面向對象軟體系統的設計而言,在支持可維護性的同時,提高系統的可復用性是一個至關重要的問題,如何同時提高一個軟體系統的可維護性和可復用性是面向對象設計需要解決的核心問題之一。在面向對象設計中,可維護性的復用是以設計原則為基礎的。每一個原則都蘊含一些面向對象設計的思想,可以從 ...
-
環境:C++ 11 + win10 IDE:Clion 2018.3 AVL平衡樹是在BST二叉查找樹的基礎上添加了平衡機制。 我們把平衡的BST認為是任一節點的左子樹和右子樹的高度差為-1,0,1中的一種情況,即不存在相差兩層及以上。 所謂平衡機制就是BST在理想情況下搜索複雜度是o(logn) ...
-
eclipse項目過多怎麼方便管理呢? 可以使用workset來進行管理。這裡的workset跟.net 也就是visual studio中的項目解決方法類似,可以將項目、類庫進行分開管理。 可以點擊project explorer中小三角進行操作 然後在彈出界面中進行操作 ...