亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

Python Web框架Flask信號機制(signals)介紹

系統(tǒng) 1723 0

信號(signals)

Flask信號(signals, or event hooking)允許特定的發(fā)送端通知訂閱者發(fā)生了什么(既然知道發(fā)生了什么,那我們可以知道接下來該做什么了)。

Flask提供了一些信號(核心信號)且其它的擴展提供更多的信號。信號是用于通知訂閱者,而不應(yīng)該鼓勵訂閱者修改數(shù)據(jù)。相關(guān)信號請查閱文檔。

信號依賴于Blinker庫。

鉤子(hooks)

Flask鉤子(通常出現(xiàn)在藍圖或應(yīng)用程序現(xiàn)存的方法中,比如一些內(nèi)置裝飾器,例如before_request)不需要Blinker庫并且允許你改變請求對象(request)或者響應(yīng)對象(response)。這些改變了應(yīng)用程序(或者藍圖)的行為。比如before_request()裝飾器。

信號看起來和鉤子做同樣的事情。然而在工作方式上它們存在不同。譬如核心的before_request()處理程序以特定的順序執(zhí)行,并且可以在返回響應(yīng)之前放棄請求。相比之下,所有的信號處理器是無序執(zhí)行的,并且不修改任何數(shù)據(jù)。

一般來說,鉤子用于改變行為(比如,身份驗證或錯誤處理),而信號用于記錄事件(比如記錄日志)。

創(chuàng)建信號

因為信號依賴于Blinker庫,請確保已經(jīng)安裝。

如果你要在自己的應(yīng)用中使用信號,你可以直接使用Blinker庫。最常見的使用情況是命名一個自定義的Namespace的信號。這也是大多數(shù)時候推薦的做法:


復(fù)制代碼 代碼如下:

from blinker import Namespace
my_signals = Namespace()


現(xiàn)在你可以像這樣創(chuàng)建新的信號:


復(fù)制代碼 代碼如下:

model_saved = my_signals.signal('model-saved')


這里使用唯一的信號名并且簡化調(diào)試??梢杂胣ame屬性來訪問信號名。

對擴展開發(fā)者:

如果你正在編寫一個Flask擴展,你想優(yōu)雅地減少缺少Blinker安裝的影響,你可以這樣做使用flask.signals.Namespace類。

訂閱信號

你可以使用信號的connect()方法來訂閱信號。第一個參數(shù)是信號發(fā)出時要調(diào)用的函數(shù),第二個參數(shù)是可選的,用于確定信號的發(fā)送者。一個信號可以擁有多個訂閱者。你可以用disconnect()方法來退訂信號。

對于所有的核心Flask信號,發(fā)送者都是發(fā)出信號的應(yīng)用。當你訂閱一個信號,請確保也提供一個發(fā)送者,除非你確實想監(jiān)聽全部應(yīng)用的信號。這在你開發(fā)一個擴展的時候尤其正確。

比如這里有一個用于在單元測試中找出哪個模板被渲染和傳入模板的變量的助手上下文管理器:


復(fù)制代碼 代碼如下:

from flask import template_rendered
from contextlib import contextmanager

@contextmanager
def captured_templates(app):
??? recorded = []
??? def record(sender, template, context, **extra):
??????? recorded.append((template, context))
??? template_rendered.connect(record, app)
??? try:
??????? yield recorded
??? finally:
??????? template_rendered.disconnect(record, app)


現(xiàn)在可以輕松地與測試客戶端配對:


復(fù)制代碼 代碼如下:

with captured_templates(app) as templates:
??? rv = app.test_client().get('/')
??? assert rv.status_code == 200
??? assert len(templates) == 1
??? template, context = templates[0]
??? assert template.name == 'index.html'
??? assert len(context['items']) == 10


確保訂閱使用了一個額外的 **extra 參數(shù),這樣當 Flask 對信號引入新參數(shù)時你的調(diào)用不會失敗。

代碼中,從 with 塊的應(yīng)用 app 中流出的渲染的所有模板現(xiàn)在會被記錄到templates 變量。無論何時模板被渲染,模板對象的上下文中添加上它。

此外,也有一個方便的助手方法(connected_to()) ,它允許你臨時地用它自己的上下文管理器把函數(shù)訂閱到信號。因為上下文管理器的返回值不能按那種方法給定,則需要把列表作為參數(shù)傳入:


復(fù)制代碼 代碼如下:

from flask import template_rendered

def captured_templates(app, recorded, **extra):
??? def record(sender, template, context):
??????? recorded.append((template, context))
??? return template_rendered.connected_to(record, app)


上面的例子看起來像這樣:


復(fù)制代碼 代碼如下:

templates = []
with captured_templates(app, templates, **extra):
??? ...
??? template, context = templates[0]

發(fā)送信號

如果你想要發(fā)送信號,你可以通過調(diào)用 send() 方法來達到目的。它接受一個發(fā)件者作為第一個參數(shù)和一些可選的被轉(zhuǎn)發(fā)到信號用戶的關(guān)鍵字參數(shù):


復(fù)制代碼 代碼如下:

class Model(object):
??? ...

??? def save(self):
??????? model_saved.send(self)


永遠嘗試選擇一個合適的發(fā)送者。如果你有一個發(fā)出信號的類,把 self 作為發(fā)送者。如果你從一個隨機的函數(shù)發(fā)出信號,把current_app._get_current_object()作為發(fā)送者。

基于信號訂閱的裝飾器

在Blinker 1.1中通過使用新的connect_via()裝飾器你也能夠輕易地訂閱信號:

復(fù)制代碼 代碼如下:

from flask import template_rendered

@template_rendered.connect_via(app)
def when_template_rendered(sender, template, context, **extra):
??? print 'Template %s is rendered with %s' % (template.name, context)

舉例

模板渲染

template_rendered信號是Flask核心信號。

當一個模版成功地渲染,這個信號會被發(fā)送。這個信號與模板實例 template 和上下文的詞典(名為 context )一起調(diào)用。

信號發(fā)送:

復(fù)制代碼 代碼如下:

def _render(template, context, app):
??? """Renders the template and fires the signal"""
??? rv = template.render(context)
??? template_rendered.send(app, template=template, context=context)
??? return rv

訂閱示例:
復(fù)制代碼 代碼如下:

def log_template_renders(sender, template, context, **extra):
??? sender.logger.debug('Rendering template "%s" with context %s',
??????????????????????? template.name or 'string template',
??????????????????????? context)

from flask import template_rendered
template_rendered.connect(log_template_renders, app)


帳號追蹤

user_logged_in是Flask-User中定義的信號,當用戶成功登入之后,這個信號會被發(fā)送。

發(fā)送信號:

復(fù)制代碼 代碼如下:

# Send user_logged_in signal
user_logged_in.send(current_app._get_current_object(), user=user)

下面這個例子追蹤登錄次數(shù)和登錄IP:
復(fù)制代碼 代碼如下:

# This code has not been tested

from flask import request
from flask_user.signals import user_logged_in

@user_logged_in.connect_via(app)
def _track_logins(sender, user, **extra):
??? user.login_count += 1
??? user.last_login_ip = request.remote_addr
??? db.session.add(user)
??? db.session.commit()

小結(jié)

信號可以讓你在一瞬間安全地訂閱它們。例如,這些臨時的訂閱對測試很有幫助。使用信號時,不要讓信號訂閱者(接收者)發(fā)生異常,因為異常會造成程序中斷。


更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯(lián)系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦?。?!

發(fā)表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 91精品国产综合久久青草 | 亚洲香蕉在线 | 99精品国产在热久久 | 国产成人精品日本亚洲麻豆 | 亚洲国产99999在线精品一区 | 久久精品视频播放 | 精品国产区 | 久国产精品视频 | 久久毛片免费看一区二区三区 | 97视频在线观看视频在线精品 | 黄片a级毛片 | 91视频青娱乐 | 欧美成人免费在线观看 | 色综合久久88色综合天天 | 国产亚洲玖玖玖在线观看 | 国产欧美亚洲精品 | 久久青草18免费观看网站 | 国产伦精品一区二区三区女 | 2019中文字幕视频 | 久久久在线视频 | 青青青在线视频人视频在线 | 国产精品2020在线看亚瑟 | 久久久久一区二区三区 | 欧美精品一区二区精品久久 | 欧美在线播放成人免费 | 最新毛片久热97免费精品视频 | 九九热精品免费 | 日日摸夜夜添夜夜添欧美毛片 | 国产中文字幕亚洲 | 日韩精品一区二区三区四区 | 中文字幕 亚洲精品 第1页 | 青青青爽国产在线视频 | 最新久久精品 | 国产精品久久久久久久久齐齐 | 99久久99热久久 | 亚洲精品国产精品乱码不97 | 久久99在线| 四虎4hu永久免费国产精品 | 国产永久一区二区三区 | 亚洲视频99 | 国产自产拍精品视频免费看 |