99999久久久久久亚洲,欧美人与禽猛交狂配,高清日韩av在线影院,一个人在线高清免费观看,啦啦啦在线视频免费观看www

熱線電話:13121318867

登錄
首頁精彩閱讀Python上下文管理器
Python上下文管理器
2017-04-18
收藏

Python上下文管理器

上下文管理器(context manager)是Python2.5開始支持的一種語法,用于規(guī)定某個對象的使用范圍。一旦進(jìn)入或者離開該使用范圍,會有特殊操作被調(diào)用 (比如為對象分配或者釋放內(nèi)存)。它的語法形式是with...as...

關(guān)閉文件
我們會進(jìn)行這樣的操作:打開文件,讀寫,關(guān)閉文件。程序員經(jīng)常會忘記關(guān)閉文件。上下文管理器可以在不需要文件的時候,自動關(guān)閉文件。
下面我們看一下兩段程序:
# without context manager
f = open("new.txt", "w")
print(f.closed)               # whether the file is open
f.write("Hello World!")
f.close()
print(f.closed)
以及:

# with context manager
with open("new.txt", "w") as f:
    print(f.closed)
    f.write("Hello World!")
print(f.closed)
兩段程序?qū)嶋H上執(zhí)行的是相同的操作。我們的第二段程序就使用了上下文管理器 (with...as...)。上下文管理器有隸屬于它的程序塊。當(dāng)隸屬的程序塊執(zhí)行結(jié)束的時候(也就是不再縮進(jìn)),上下文管理器自動關(guān)閉了文件 (我們通過f.closed來查詢文件是否關(guān)閉)。我們相當(dāng)于使用縮進(jìn)規(guī)定了文件對象f的使用范圍。
上面的上下文管理器基于f對象的__exit__()特殊方法(還記得我們?nèi)绾卫锰厥夥椒▉韺崿F(xiàn)各種語法?參看特殊方法與多范式)。當(dāng)我們使用上下文管理器的語法時,我們實際上要求Python在進(jìn)入程序塊之前調(diào)用對象的__enter__()方法,在結(jié)束程序塊的時候調(diào)用__exit__()方法。對于文件對象f來說,它定義了__enter__()和__exit__()方法(可以通過dir(f)看到)。在f的__exit__()方法中,有self.close()語句。所以在使用上下文管理器時,我們就不用明文關(guān)閉f文件了。
自定義
任何定義了__enter__()和__exit__()方法的對象都可以用于上下文管理器。文件對象f是內(nèi)置對象,所以f自動帶有這兩個特殊方法,不需要自定義。
下面,我們自定義用于上下文管理器的對象,就是下面的myvow:
# customized object

class VOW(object):
    def __init__(self, text):
        self.text = text
    def __enter__(self):
        self.text = "I say: " + self.text    # add prefix
        return self                          # note: return an object
    def __exit__(self,exc_type,exc_value,traceback):
        self.text = self.text + "!"          # add suffix

with VOW("I'm fine") as myvow:
    print(myvow.text)

print(myvow.text)
我們的運行結(jié)果如下:
I say: I'm fine
I say: I'm fine!
我們可以看到,在進(jìn)入上下文和離開上下文時,對象的text屬性發(fā)生了改變(最初的text屬性是"I'm fine")。
__enter__()返回一個對象。上下文管理器會使用這一對象作為as所指的變量,也就是myvow。在__enter__()中,我們?yōu)閙yvow.text增加了前綴 ("I say: ")。在__exit__()中,我們?yōu)閙yvow.text增加了后綴("!")。
注意: __exit__()中有四個參數(shù)。當(dāng)程序塊中出現(xiàn)異常(exception),__exit__()的參數(shù)中exc_type, exc_value, traceback用于描述異常。我們可以根據(jù)這三個參數(shù)進(jìn)行相應(yīng)的處理。如果正常運行結(jié)束,這三個參數(shù)都是None。在我們的程序中,我們并沒有用到這一特性。數(shù)據(jù)分析師培訓(xùn)
總結(jié):
通過上下文管理器,我們控制對象在程序不同區(qū)間的特性。上下文管理器(with EXPR as VAR)大致相當(dāng)于如下流程:
# with EXPR as VAR:

VAR = EXPR
VAR = VAR.__enter__()
try:
    BLOCK
finally:
    VAR.__exit__()
由于上下文管理器帶來的便利,它是一個值得使用的工具。

數(shù)據(jù)分析咨詢請掃描二維碼

若不方便掃碼,搜微信號:CDAshujufenxi

數(shù)據(jù)分析師資訊
更多

OK
客服在線
立即咨詢
客服在線
立即咨詢
') } function initGt() { var handler = function (captchaObj) { captchaObj.appendTo('#captcha'); captchaObj.onReady(function () { $("#wait").hide(); }).onSuccess(function(){ $('.getcheckcode').removeClass('dis'); $('.getcheckcode').trigger('click'); }); window.captchaObj = captchaObj; }; $('#captcha').show(); $.ajax({ url: "/login/gtstart?t=" + (new Date()).getTime(), // 加隨機(jī)數(shù)防止緩存 type: "get", dataType: "json", success: function (data) { $('#text').hide(); $('#wait').show(); // 調(diào)用 initGeetest 進(jìn)行初始化 // 參數(shù)1:配置參數(shù) // 參數(shù)2:回調(diào),回調(diào)的第一個參數(shù)驗證碼對象,之后可以使用它調(diào)用相應(yīng)的接口 initGeetest({ // 以下 4 個配置參數(shù)為必須,不能缺少 gt: data.gt, challenge: data.challenge, offline: !data.success, // 表示用戶后臺檢測極驗服務(wù)器是否宕機(jī) new_captcha: data.new_captcha, // 用于宕機(jī)時表示是新驗證碼的宕機(jī) product: "float", // 產(chǎn)品形式,包括:float,popup width: "280px", https: true // 更多配置參數(shù)說明請參見:http://docs.geetest.com/install/client/web-front/ }, handler); } }); } function codeCutdown() { if(_wait == 0){ //倒計時完成 $(".getcheckcode").removeClass('dis').html("重新獲取"); }else{ $(".getcheckcode").addClass('dis').html("重新獲取("+_wait+"s)"); _wait--; setTimeout(function () { codeCutdown(); },1000); } } function inputValidate(ele,telInput) { var oInput = ele; var inputVal = oInput.val(); var oType = ele.attr('data-type'); var oEtag = $('#etag').val(); var oErr = oInput.closest('.form_box').next('.err_txt'); var empTxt = '請輸入'+oInput.attr('placeholder')+'!'; var errTxt = '請輸入正確的'+oInput.attr('placeholder')+'!'; var pattern; if(inputVal==""){ if(!telInput){ errFun(oErr,empTxt); } return false; }else { switch (oType){ case 'login_mobile': pattern = /^1[3456789]\d{9}$/; if(inputVal.length==11) { $.ajax({ url: '/login/checkmobile', type: "post", dataType: "json", data: { mobile: inputVal, etag: oEtag, page_ur: window.location.href, page_referer: document.referrer }, success: function (data) { } }); } break; case 'login_yzm': pattern = /^\d{6}$/; break; } if(oType=='login_mobile'){ } if(!!validateFun(pattern,inputVal)){ errFun(oErr,'') if(telInput){ $('.getcheckcode').removeClass('dis'); } }else { if(!telInput) { errFun(oErr, errTxt); }else { $('.getcheckcode').addClass('dis'); } return false; } } return true; } function errFun(obj,msg) { obj.html(msg); if(msg==''){ $('.login_submit').removeClass('dis'); }else { $('.login_submit').addClass('dis'); } } function validateFun(pat,val) { return pat.test(val); }