Flask框架中的信号基于blinker,其主要就是让开发者可是在flask请求过程中定制一些用户行为。

  1. pip3 install blinker

1. 内置信号

  1. request_started = _signals.signal('request-started') # 请求到来前执行
  2. request_finished = _signals.signal('request-finished') # 请求结束后执行
  3.  
  4. before_render_template = _signals.signal('before-render-template') # 模板渲染前执行
  5. template_rendered = _signals.signal('template-rendered') # 模板渲染后执行
  6.  
  7. got_request_exception = _signals.signal('got-request-exception') # 请求执行出现异常时执行
  8.  
  9. request_tearing_down = _signals.signal('request-tearing-down') # 请求执行完毕后自动执行(无论成功与否)
  10. appcontext_tearing_down = _signals.signal('appcontext-tearing-down')# 请求上下文执行完毕后自动执行(无论成功与否)
  11.  
  12. appcontext_pushed = _signals.signal('appcontext-pushed') # 请求上下文push时执行
  13. appcontext_popped = _signals.signal('appcontext-popped') # 请求上下文pop时执行
  14. message_flashed = _signals.signal('message-flashed') # 调用flask在其中添加数据时,自动触发

  

2. 源码示例

  1. class Flask(_PackageBoundObject):
  2.  
  3. def full_dispatch_request(self):
  4.  
  5. self.try_trigger_before_first_request_functions()
  6. try:
  7. # ############### 触发request_started 信号 ###############
  8. request_started.send(self)
  9. rv = self.preprocess_request()
  10. if rv is None:
  11. rv = self.dispatch_request()
  12. except Exception as e:
  13. rv = self.handle_user_exception(e)
  14. response = self.make_response(rv)
  15. response = self.process_response(response)
  16.  
  17. # ############### request_finished 信号 ###############
  18. request_finished.send(self, response=response)
  19. return response
  20.  
  21. def wsgi_app(self, environ, start_response):
  22.  
  23. ctx = self.request_context(environ)
  24. ctx.push()
  25. error = None
  26. try:
  27. try:
  28. response = self.full_dispatch_request()
  29. except Exception as e:
  30. error = e
  31. response = self.make_response(self.handle_exception(e))
  32. return response(environ, start_response)
  33. finally:
  34. if self.should_ignore_error(error):
  35. error = None
  36. ctx.auto_pop(error)

request_started

  1. 同上

request_finished

  1. def render_template(template_name_or_list, **context):
  2. """Renders a template from the template folder with the given
  3. context.
  4.  
  5. :param template_name_or_list: the name of the template to be
  6. rendered, or an iterable with template names
  7. the first one existing will be rendered
  8. :param context: the variables that should be available in the
  9. context of the template.
  10. """
  11. ctx = _app_ctx_stack.top
  12. ctx.app.update_template_context(context)
  13. return _render(ctx.app.jinja_env.get_or_select_template(template_name_or_list),
  14. context, ctx.app)
  15.  
  16. def _render(template, context, app):
  17. """Renders the template and fires the signal"""
  18.  
  19. # ############### before_render_template 信号 ###############
  20. before_render_template.send(app, template=template, context=context)
  21. rv = template.render(context)
  22.  
  23. # ############### template_rendered 信号 ###############
  24. template_rendered.send(app, template=template, context=context)
  25. return rv

before_render_template

  1. 同上

template_rendered

  1. class Flask(_PackageBoundObject):
  2.  
  3. def handle_exception(self, e):
  4.  
  5. exc_type, exc_value, tb = sys.exc_info()
  6.  
  7. # ############### got_request_exception 信号 ###############
  8. got_request_exception.send(self, exception=e)
  9. handler = self._find_error_handler(InternalServerError())
  10.  
  11. if self.propagate_exceptions:
  12. # if we want to repropagate the exception, we can attempt to
  13. # raise it with the whole traceback in case we can do that
  14. # (the function was actually called from the except part)
  15. # otherwise, we just raise the error again
  16. if exc_value is e:
  17. reraise(exc_type, exc_value, tb)
  18. else:
  19. raise e
  20.  
  21. self.log_exception((exc_type, exc_value, tb))
  22. if handler is None:
  23. return InternalServerError()
  24. return handler(e)
  25.  
  26. def wsgi_app(self, environ, start_response):
  27.  
  28. ctx = self.request_context(environ)
  29. ctx.push()
  30. error = None
  31. try:
  32. try:
  33. response = self.full_dispatch_request()
  34. except Exception as e:
  35. error = e
  36. # 这里这里这里这里这里这里这里这里这里这里这里这里 #
  37. response = self.make_response(self.handle_exception(e))
  38. return response(environ, start_response)
  39. finally:
  40. if self.should_ignore_error(error):
  41. error = None
  42. ctx.auto_pop(error)

got_request_exception

  1. class AppContext(object):
  2. def push(self):
  3. """Binds the app context to the current context."""
  4. self._refcnt += 1
  5. if hasattr(sys, 'exc_clear'):
  6. sys.exc_clear()
  7. _app_ctx_stack.push(self)
  8. # ############## 触发 appcontext_pushed 信号 ##############
  9. appcontext_pushed.send(self.app)
  10.  
  11. def pop(self, exc=_sentinel):
  12. """Pops the app context."""
  13. try:
  14. self._refcnt -= 1
  15. if self._refcnt <= 0:
  16. if exc is _sentinel:
  17. exc = sys.exc_info()[1]
  18. # ############## 触发 appcontext_tearing_down 信号 ##############
  19. self.app.do_teardown_appcontext(exc)
  20. finally:
  21. rv = _app_ctx_stack.pop()
  22. assert rv is self, 'Popped wrong app context. (%r instead of %r)' \
  23. % (rv, self)
  24.  
  25. # ############## 触发 appcontext_popped 信号 ##############
  26. appcontext_popped.send(self.app)
  27.  
  28. class RequestContext(object):
  29. def push(self):
  30. top = _request_ctx_stack.top
  31. if top is not None and top.preserved:
  32. top.pop(top._preserved_exc)
  33.  
  34. app_ctx = _app_ctx_stack.top
  35. if app_ctx is None or app_ctx.app != self.app:
  36.  
  37. # ####################################################
  38. app_ctx = self.app.app_context()
  39. app_ctx.push()
  40. self._implicit_app_ctx_stack.append(app_ctx)
  41. else:
  42. self._implicit_app_ctx_stack.append(None)
  43.  
  44. if hasattr(sys, 'exc_clear'):
  45. sys.exc_clear()
  46.  
  47. _request_ctx_stack.push(self)
  48.  
  49. # Open the session at the moment that the request context is
  50. # available. This allows a custom open_session method to use the
  51. # request context (e.g. code that access database information
  52. # stored on `g` instead of the appcontext).
  53. self.session = self.app.open_session(self.request)
  54. if self.session is None:
  55. self.session = self.app.make_null_session()
  56.  
  57. class Flask(_PackageBoundObject):
  58.  
  59. def wsgi_app(self, environ, start_response):
  60.  
  61. ctx = self.request_context(environ)
  62. ctx.push()
  63. error = None
  64. try:
  65. try:
  66. response = self.full_dispatch_request()
  67. except Exception as e:
  68. error = e
  69. response = self.make_response(self.handle_exception(e))
  70. return response(environ, start_response)
  71. finally:
  72. if self.should_ignore_error(error):
  73. error = None
  74. ctx.auto_pop(error)
  75.  
  76. def pop(self, exc=_sentinel):
  77. app_ctx = self._implicit_app_ctx_stack.pop()
  78.  
  79. try:
  80. clear_request = False
  81. if not self._implicit_app_ctx_stack:
  82. self.preserved = False
  83. self._preserved_exc = None
  84. if exc is _sentinel:
  85. exc = sys.exc_info()[1]
  86.  
  87. # ################## 触发 request_tearing_down 信号 ##################
  88. self.app.do_teardown_request(exc)
  89.  
  90. # If this interpreter supports clearing the exception information
  91. # we do that now. This will only go into effect on Python 2.x,
  92. # on 3.x it disappears automatically at the end of the exception
  93. # stack.
  94. if hasattr(sys, 'exc_clear'):
  95. sys.exc_clear()
  96.  
  97. request_close = getattr(self.request, 'close', None)
  98. if request_close is not None:
  99. request_close()
  100. clear_request = True
  101. finally:
  102. rv = _request_ctx_stack.pop()
  103.  
  104. # get rid of circular dependencies at the end of the request
  105. # so that we don't require the GC to be active.
  106. if clear_request:
  107. rv.request.environ['werkzeug.request'] = None
  108.  
  109. # Get rid of the app as well if necessary.
  110. if app_ctx is not None:
  111. # ####################################################
  112. app_ctx.pop(exc)
  113.  
  114. assert rv is self, 'Popped wrong request context. ' \
  115. '(%r instead of %r)' % (rv, self)
  116.  
  117. def auto_pop(self, exc):
  118. if self.request.environ.get('flask._preserve_context') or \
  119. (exc is not None and self.app.preserve_context_on_exception):
  120. self.preserved = True
  121. self._preserved_exc = exc
  122. else:
  123. self.pop(exc)

request_tearing_down

  1. 同上

request_tearing_down

  1. 同上

appcontext_tearing_down

  1. 同上

appcontext_pushed

  1. 同上

appcontext_popped

  1. def flash(message, category='message'):
  2. """Flashes a message to the next request. In order to remove the
  3. flashed message from the session and to display it to the user,
  4. the template has to call :func:`get_flashed_messages`.
  5.  
  6. .. versionchanged:: 0.3
  7. `category` parameter added.
  8.  
  9. :param message: the message to be flashed.
  10. :param category: the category for the message. The following values
  11. are recommended: ``'message'`` for any kind of message,
  12. ``'error'`` for errors, ``'info'`` for information
  13. messages and ``'warning'`` for warnings. However any
  14. kind of string can be used as category.
  15. """
  16. # Original implementation:
  17. #
  18. # session.setdefault('_flashes', []).append((category, message))
  19. #
  20. # This assumed that changes made to mutable structures in the session are
  21. # are always in sync with the session object, which is not true for session
  22. # implementations that use external storage for keeping their keys/values.
  23. flashes = session.get('_flashes', [])
  24. flashes.append((category, message))
  25. session['_flashes'] = flashes
  26.  
  27. # ############### 触发 message_flashed 信号 ###############
  28. message_flashed.send(current_app._get_current_object(),
  29. message=message, category=category)

message_flashed

3. 执行流程

  1. a. before_first_request
  2. b. 触发 request_started 信号
  3. c. before_request
  4. d. 模板渲染
  5. 渲染前的信号 before_render_template.send(app, template=template, context=context)
  6. rv = template.render(context) # 模板渲染
  7. 渲染后的信号 template_rendered.send(app, template=template, context=context)
  8. e. after_request
  9. f. session.save_session()
  10. g. 触发 request_finished信号
  11.  
  12. 如果上述过程出错:
  13. 触发错误处理信号 got_request_exception.send(self, exception=e)
  14.  
  15. h. 触发信号 request_tearing_down

  

4. 自定义信号

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3. from flask import Flask, current_app, flash, render_template
  4. from flask.signals import _signals
  5.  
  6. app = Flask(import_name=__name__)
  7.  
  8. # 自定义信号
  9. xxxxx = _signals.signal('xxxxx')
  10.  
  11. def func(sender, *args, **kwargs):
  12. print(sender)
  13.  
  14. # 自定义信号中注册函数
  15. xxxxx.connect(func)
  16.  
  17. @app.route("/x")
  18. def index():
  19. # 触发信号
  20. xxxxx.send('123123', k1='v1')
  21. return 'Index'
  22.  
  23. if __name__ == '__main__':
  24. app.run()

  

了解Flask 信号机制的更多相关文章

  1. Inside Flask - signal 信号机制

    Inside Flask - signal 信号机制 singal 在平常的 flask web 开发过程中较少接触到,但对于使用 flask 进行框架级别的开发时,则必须了解相关的工作机制.flas ...

  2. Flask笔记:信号机制

    Flask中有内置的一些信号,也可以通过三方库blinker自定义信号,其实Flask内置的信号也是优先使用的blinker库,如果没有安装blinker才会使用自定义的信号机制.可以通过点击任意导入 ...

  3. 信号(Django信号、Flask信号、Scrapy信号)

    简介 Django.Flask.scrapy都包含了一个“信号分配器”,使得当一些动作在框架的其他地方发生的时候,解耦的应用可以得到提醒. 通俗来讲,就是一些动作发生的时候,信号允许特定的发送者去提醒 ...

  4. linux信号机制与python信号量

    1.信号本质 软中断信号(signal,又简称为信号)用来通知进程发生了异步事件.在软件层次上是对中断机制的一种模拟,在原理上,一个进程收到一个信号与处理器收到一个中断请求可以说是一样的.信号是进程间 ...

  5. linux信号机制 - 用户堆栈和内核堆栈的变化【转】

    转自:http://itindex.net/detail/16418-linux-%E4%BF%A1%E5%8F%B7-%E5%A0%86%E6%A0%88 此文只简单分析发送信号给用户程序后,用户堆 ...

  6. 利用linux信号机制调试段错误(Segment fault)

    在实际开发过程中,大家可能会遇到段错误的问题,虽然是个老问题,但是其带来的隐患是极大的,只要出现一次,程序立即崩溃中止.如果程序运行在PC中,segment fault的调试相对比较方便,因为可以通过 ...

  7. Django的信号机制

    Django提供一种信号机制.其实就是观察者模式,又叫发布-订阅(Publish/Subscribe) .当发生一些动作的时候,发出信号,然后监听了这个信号的callback函数就会执行. Djang ...

  8. linux中的信号机制

    概述 Linux信号机制是在应用软件层次上对中断机制的一种模拟,信号提供了一种处理异步事件的方法,例如,终端用户输入中断键(ctrl+c),则会通过信号机制停止一个程序[1]. 这其实就是向那个程序( ...

  9. flask信号使用

    flask信号: 安装: flask中的信号使用的是一个第三方插件,叫做blinker.通过pip list看一下,如果没有安装,通过以下命令即可安装blinker: pip install blin ...

随机推荐

  1. Java输入输出流(2)

    6. Java.IO流类库 1. io流的四个基本类 java.io包中包括了流式I/O所须要的全部类. 在java.io包中有四个基本类:InputStream.OutputStream及Reade ...

  2. os.path模块【python】

    os.path.abspath(path) #返回绝对路径 os.path.basename(path) #返回文件名 os.path.commonprefix(list) #返回list(多个路径) ...

  3. js方法随机抽取n个随机数

    function getImageRandomPosition(){ do { var n = Math.floor(Math.random() * 12);//n为随机出现的0-11之内的数值 fo ...

  4. Gamma编码及Delta编码概述

    一.Elias Gamma Coding 即Gamma编码,是一种对正整数进行编码的统一编码,由Peter Elias发明.适用于预先无法获知最大编码整数的情况,而且小整数出现频率高,大整数出现频率低 ...

  5. 《转》python学习(6)序列类型-字符串

    转自 http://www.cnblogs.com/BeginMan/archive/2013/06/08/3125502.html 二.序列类型 包含字符串.列表.元祖.模式都一样,举一反三即可.如 ...

  6. python初学总结(二)

    (1)字典 字典是一种映射关系:键(key),值(value),key-value对 创建字典的方式:直接创建和利用dict函数创建 >>> aInfo = {'Wangdachui ...

  7. 解决提示“配色方案已更改为Windows7 Basic”

    WIN7是很多用户都用过的系统,是由微软推出的.下面就说一个小技巧. 如何解决Win7系统提示:“配色方案已更改为Windows 7 Basic”解决方案.   更改Win7配色方案 首先,右击桌面空 ...

  8. 【BZOJ2661】[BeiJing wc2012]连连看 最大费用流

    [BZOJ2661][BeiJing wc2012]连连看 Description 凡是考智商的题里面总会有这么一种消除游戏.不过现在面对的这关连连看可不是QQ游戏里那种考眼力的游戏.我们的规则是,给 ...

  9. svn-maven-tomcat自动发布脚本

    #!/bin/sh #svn-maven-tomcat自动发布脚本 #变量设置 svnpath=svn://10.60.10.120/研发部/xx-maven svnusername=xxx svnp ...

  10. PHP中文字数限制:中文字符串截取(mb_substr)

    一.中文截取:mb_substr() mb_substr( $str, $start, $length, $encoding ) $str,需要截断的字符串 $start,截断开始处,起始处为0 $l ...