Flask的核心機(jī)制恰响!關(guān)于請(qǐng)求處理流程和上下文

學(xué)習(xí)一樣?xùn)|西不能只停留在表面,我們要探索其中的細(xì)節(jié)魁亦,學(xué)習(xí)作者的編程思想渔隶,這樣才能更進(jìn)一步。

關(guān)于WSGI

WSGI(全稱(chēng)Web Server Gateway Interface),是為 Python 語(yǔ)言定義的Web服務(wù)器Web應(yīng)用程序之間的一種簡(jiǎn)單而通用的接口间唉,它封裝了接受HTTP請(qǐng)求绞灼、解析HTTP請(qǐng)求發(fā)送HTTP呈野,響應(yīng)等等的這些底層的代碼和操作低矮,使開(kāi)發(fā)者可以高效的編寫(xiě)Web應(yīng)用。

一個(gè)簡(jiǎn)單的使用WSGI的App例子:

def application(environ, start_response): 
    start_response('200 OK', [('Content-Type', 'text/html')]) 
    return [b'<h1>Hello, I Am WSGI!</h1>']
  • environ: 一個(gè)包含全部HTTP請(qǐng)求信息的字典被冒,由WSGI Server解包HTTP請(qǐng)求生成军掂。
  • start_response: 一個(gè)WSGI Server提供的函數(shù),調(diào)用可以發(fā)送響應(yīng)的狀態(tài)碼和HTTP報(bào)文頭昨悼, 函數(shù)在返回前必須調(diào)用一次start_response()蝗锥。
  • application()應(yīng)當(dāng)返回一個(gè)可以迭代的對(duì)象(HTTP正文)。
  • application()函數(shù)由WSGI Server直接調(diào)用和提供參數(shù)率触。
  • Python內(nèi)置了一個(gè)WSGIREFWSGI Server终议,不過(guò)性能不是很好,一般只用在開(kāi)發(fā)環(huán)境葱蝗⊙ㄕ牛可以選擇其他的如Gunicorn
WSGI Server 和 App交互圖

Flask的上下文對(duì)象

Flask有兩種Context(上下文)两曼,分別是

  • RequestContext 請(qǐng)求上下文
  • Request 請(qǐng)求的對(duì)象皂甘,封裝了Http請(qǐng)求(environ)的內(nèi)容
  • Session 根據(jù)請(qǐng)求中的cookie,重新載入該訪(fǎng)問(wèn)者相關(guān)的會(huì)話(huà)信息悼凑。
  • AppContext 程序上下文
  • g 處理請(qǐng)求時(shí)用作臨時(shí)存儲(chǔ)的對(duì)象偿枕。每次請(qǐng)求都會(huì)重設(shè)這個(gè)變量
  • current_app 當(dāng)前激活程序的程序?qū)嵗?/li>

生命周期:

  • current_app的生命周期最長(zhǎng),只要當(dāng)前程序?qū)嵗€在運(yùn)行佛析,都不會(huì)失效益老。
  • Requestg的生命周期為一次請(qǐng)求期間,當(dāng)請(qǐng)求處理完成后寸莫,生命周期也就完結(jié)了
  • Session就是傳統(tǒng)意義上的session了捺萌。只要它還未失效(用戶(hù)未關(guān)閉瀏覽器、沒(méi)有超過(guò)設(shè)定的失效時(shí)間)膘茎,那么不同的請(qǐng)求會(huì)共用同樣的session桃纯。

Flask處理流程

Flask處理請(qǐng)求流程
  • 第一步:創(chuàng)建上下文
    Flask根據(jù)WSGI Server封裝的請(qǐng)求等的信息(environ)新建RequestContext對(duì)象AppContext對(duì)象
# 聲明對(duì)象
# LocalStack  LocalProxy 都由Werkzeug提供
# 我們不深究他的細(xì)節(jié),那又是另外一個(gè)故事了披坏,我們只需知道他的作用就行了
# LocalStack 是棧結(jié)構(gòu)态坦,可以將對(duì)象推入、彈出
# 也可以快速拿到棧頂對(duì)象棒拂。當(dāng)然伞梯,所有的修改都只在本線(xiàn)程可見(jiàn)玫氢。
_request_ctx_stack = LocalStack()
_app_ctx_stack = LocalStack()
# 如果調(diào)用一個(gè)LocalStack實(shí)例, 能返回一個(gè) LocalProxy 對(duì)象
# 這個(gè)對(duì)象始終指向 這個(gè)LocalStack實(shí)例的棧頂元素谜诫。
# 如果棧頂元素不存在漾峡,訪(fǎng)問(wèn)這個(gè) LocalProxy 的時(shí)候會(huì)拋出 RuntimeError異常
# LocalProxy對(duì)象你只需暫時(shí)理解為棧里面的元素即可了
current_app = LocalProxy(_find_app)
request = LocalProxy(partial(_lookup_req_object, 'request'))
session = LocalProxy(partial(_lookup_req_object, 'session'))
g = LocalProxy(partial(_lookup_app_object, 'g'))
# RequestContext
class RequestContext(object):
    def __init__(self, app, environ, request=None):    
        self.app = app    
        if request is None:        
                request = app.request_class(environ)    
        self.request = request    
        self.url_adapter = app.create_url_adapter(self.request)    
        self.flashes = None    
        self.session = None
#AppContext
class AppContext(object):
    def __init__(self, app):    
        self.app = app    
        self.url_adapter = app.create_url_adapter(None)    
        self.g = app.app_ctx_globals_class()    
        self._refcnt = 0

這里需要注意的是,RequestContext在初始化的時(shí)候喻旷,當(dāng)前Flask的實(shí)例作為參數(shù)被傳進(jìn)來(lái)生逸。雖然每次的請(qǐng)求處理都會(huì)創(chuàng)建一個(gè)RequestContext對(duì)象,但是每一次傳入的app參數(shù)卻是同一個(gè)且预。通過(guò)這個(gè)機(jī)制槽袄,可以使得:

由同一個(gè)Flask實(shí)例所創(chuàng)建的RequestContext,其成員變量app都是同一個(gè)Flask實(shí)例對(duì)象 锋谐。實(shí)現(xiàn)了多個(gè)RequestContext對(duì)應(yīng)同一個(gè)current_app 的目的遍尺。

  • 第二步:入棧
    RequestContext對(duì)象push進(jìn)_request_ctx_stack里面。在這次請(qǐng)求期間涮拗,訪(fǎng)問(wèn)request對(duì)象狮鸭,session對(duì)象將指向這個(gè)棧的棧頂元素
class RequestContext(object):
    def push(self):   
        ....
        _app_ctx_stack.push(self)   
        appcontext_pushed.send(self.app)

AppContext對(duì)象push進(jìn)_app_ctx_stack里面。在這次請(qǐng)求期間多搀,訪(fǎng)問(wèn)g對(duì)象將指向這個(gè)棧的棧頂元素

class AppContext(object):
    def push(self):   
        ....
        _request_ctx_stack.push(self)
  • 第三步:請(qǐng)求分發(fā)
    response = self.full_dispatch_request()
    Flask將調(diào)用full_dispatch_request函數(shù)進(jìn)行請(qǐng)求的分發(fā),之所以不用給參數(shù)灾部,是因?yàn)槲覀兛梢酝ㄟ^(guò)request對(duì)象獲得這次請(qǐng)求的信息康铭。full_dispatch_request將根據(jù)請(qǐng)求的url找到對(duì)應(yīng)的藍(lán)本里面的視圖函數(shù),并生成一個(gè)response對(duì)象赌髓。注意的是从藤,在請(qǐng)求之外的時(shí)間,訪(fǎng)問(wèn)request對(duì)象是無(wú)效的锁蠕,因?yàn)閞equest對(duì)象依賴(lài)請(qǐng)求期間的_request_ctx_stack棧夷野。

  • 第四步:上下文對(duì)象出棧
    這次HTTP的響應(yīng)已經(jīng)生成了,就不需要兩個(gè)上下文對(duì)象了荣倾。分別將兩個(gè)上下文對(duì)象出棧悯搔,為下一次的HTTP請(qǐng)求做出準(zhǔn)備。

  • 第五步:響應(yīng)WSGI
    調(diào)用Response對(duì)象舌仍,向WSGI Server返回其結(jié)果作為HTTP正文妒貌。Response對(duì)象是一個(gè) 可調(diào)用對(duì)象,當(dāng)調(diào)用發(fā)生時(shí)铸豁,將首先執(zhí)行WSGI服務(wù)器傳入的start_response()函數(shù) 發(fā)送狀態(tài)碼和HTTP報(bào)文頭灌曙。

最后附上Flask處理請(qǐng)求的wsgi_app函數(shù)

# environ: WSGI Server封裝的HTTP請(qǐng)求信息
# start_response: WSGI Server提供的函數(shù),調(diào)用可以發(fā)送狀態(tài)碼和HTTP報(bào)文頭
def wsgi_app(self, environ, start_response):
    # 根據(jù)environ創(chuàng)建上下文
    ctx = self.request_context(environ)
    # 把當(dāng)前的request context,app context綁定到當(dāng)前的context
    ctx.push()
    error = None
    try:
        try:
            #根據(jù)請(qǐng)求的URL节芥,分發(fā)請(qǐng)求在刺,經(jīng)過(guò)視圖函數(shù)處理后返回響應(yīng)對(duì)象
            response = self.full_dispatch_request()    
        except Exception as e:        
            error = e        
            response = self.make_response(self.handle_exception(e))   
        return response(environ, start_response)
    finally:   
        if self.should_ignore_error(error):        
          error = None    
       # 最后出棧
        ctx.auto_pop(error)

參考資料
匯智網(wǎng):Flask框架-上下文對(duì)象 :Flask核心機(jī)制

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子蚣驼,更是在濱河造成了極大的恐慌魄幕,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,427評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件隙姿,死亡現(xiàn)場(chǎng)離奇詭異梅垄,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)输玷,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,551評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)队丝,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人欲鹏,你說(shuō)我怎么就攤上這事机久。” “怎么了赔嚎?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,747評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵膘盖,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我尤误,道長(zhǎng)侠畔,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,939評(píng)論 1 295
  • 正文 為了忘掉前任损晤,我火速辦了婚禮软棺,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘尤勋。我一直安慰自己喘落,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,955評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布最冰。 她就那樣靜靜地躺著瘦棋,像睡著了一般。 火紅的嫁衣襯著肌膚如雪暖哨。 梳的紋絲不亂的頭發(fā)上赌朋,一...
    開(kāi)封第一講書(shū)人閱讀 51,737評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音鹿蜀,去河邊找鬼箕慧。 笑死,一個(gè)胖子當(dāng)著我的面吹牛茴恰,可吹牛的內(nèi)容都是我干的颠焦。 我是一名探鬼主播,決...
    沈念sama閱讀 40,448評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼往枣,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼伐庭!你這毒婦竟也來(lái)了粉渠?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,352評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤圾另,失蹤者是張志新(化名)和其女友劉穎霸株,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體集乔,經(jīng)...
    沈念sama閱讀 45,834評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡去件,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,992評(píng)論 3 338
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了扰路。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片尤溜。...
    茶點(diǎn)故事閱讀 40,133評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖汗唱,靈堂內(nèi)的尸體忽然破棺而出宫莱,到底是詐尸還是另有隱情,我是刑警寧澤哩罪,帶...
    沈念sama閱讀 35,815評(píng)論 5 346
  • 正文 年R本政府宣布授霸,位于F島的核電站,受9級(jí)特大地震影響际插,放射性物質(zhì)發(fā)生泄漏碘耳。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,477評(píng)論 3 331
  • 文/蒙蒙 一框弛、第九天 我趴在偏房一處隱蔽的房頂上張望藏畅。 院中可真熱鬧,春花似錦功咒、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,022評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至幽七,卻和暖如春景殷,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背澡屡。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,147評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工猿挚, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人驶鹉。 一個(gè)月前我還...
    沈念sama閱讀 48,398評(píng)論 3 373
  • 正文 我出身青樓绩蜻,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親室埋。 傳聞我的和親對(duì)象是個(gè)殘疾皇子办绝,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,077評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • Flask框架中有許多魔法將Web應(yīng)用開(kāi)發(fā)者與一些細(xì)節(jié)隔離開(kāi)來(lái)伊约,其中Context機(jī)制又是有別于其他框架的,這個(gè)機(jī)...
    靡不有初LB閱讀 1,986評(píng)論 2 5
  • 上下文這個(gè)概念多見(jiàn)于文章中孕蝉,是一句話(huà)中的語(yǔ)境屡律,也就是語(yǔ)言環(huán)境。一句莫名其妙的話(huà)出現(xiàn)會(huì)讓人不理解什么意思降淮,如果有語(yǔ)言...
    饅頭白啊白閱讀 31,388評(píng)論 6 66
  • [TOC]一直想做源碼閱讀這件事超埋,總感覺(jué)難度太高時(shí)間太少,可望不可見(jiàn)佳鳖。最近正好時(shí)間充裕霍殴,決定試試做一下,并記錄一下...
    何柯君閱讀 7,190評(píng)論 3 98
  • 22年12月更新:個(gè)人網(wǎng)站關(guān)停腋颠,如果仍舊對(duì)舊教程有興趣參考 Github 的markdown內(nèi)容[https://...
    tangyefei閱讀 35,184評(píng)論 22 257
  • 回顧一下Flask的流程: WSGI Server 到 WSGI App 圖中可以看到HTTP請(qǐng)求都是通過(guò)WSGI...
    yiludege閱讀 3,437評(píng)論 0 3