1. 信号(源码)

  • 信号,是在flask框架中为我们预留的钩子,让我们可以进行一些自定义操作。

    pip3 install blinker

2. 根据flask项目的请求流程来进行设置扩展点

  • 中间件

    # 代码示例
    
    from flask import Flask,render_template
    
    app = Flask(__name__)
    
    @app.route('/index')
    def index():
    return render_template('index.html') @app.route('/order')
    def order():
    return render_template('order.html') class MyMiddleware(object):
    def __init__(self,old_app):
    self.wsgi_app = old_app.wsgi_app def __call__(self, *args, **kwargs):
    print('123')
    result = self.wsgi_app(*args, **kwargs)
    print('456')
    return result app.wsgi_app = MyMiddleware(app) if __name__ == '__main__':
    app.run()
  • 当app_ctx被push到local中栈之后,会触发appcontext_pushed信号,之前注册在这个信号中的方法,就会被执行。

    # 代码示例
    
    from flask import Flask,render_template
    from flask import signals app = Flask(__name__) @signals.appcontext_pushed.connect
    def f1(arg):
    print('appcontext_pushed信号f1被触发',arg) @signals.appcontext_pushed.connect
    def f2(arg):
    print('appcontext_pushed信号f2被触发',arg) @app.route('/index')
    def index():
    return render_template('index.html') @app.route('/order')
    def order():
    return render_template('order.html') if __name__ == '__main__':
    app.run()
    # app.__call__
  • 执行before_first_request扩展

    # 代码示例
    
    from flask import Flask,render_template
    
    app = Flask(__name__)
    
    @app.before_first_request
    def f2():
    print('before_first_requestf2被触发') @app.route('/index')
    def index():
    return render_template('index.html') @app.route('/order')
    def order():
    return render_template('order.html') if __name__ == '__main__':
    app.run()
  • request_started信号

    # 代码示例
    
    from flask import Flask,render_template
    from flask import signals
    app = Flask(__name__) @signals.request_started.connect
    def f3(arg):
    print('request_started信号被触发',arg) @app.route('/index')
    def index():
    return render_template('index.html') @app.route('/order')
    def order():
    return render_template('order.html') if __name__ == '__main__':
    app.run()
  • url_value_processor

    # 代码示例
    
    from flask import Flask,render_template,g
    from flask import signals
    app = Flask(__name__) @app.url_value_preprocessor
    def f5(endpoint,args):
    print('f5') @app.route('/index/')
    def index():
    print('index')
    return render_template('index.html') @app.route('/order')
    def order():
    print('order')
    return render_template('order.html') if __name__ == '__main__':
    app.run()
  • before_reuqest

    # 代码示例
    
    from flask import Flask,render_template,g
    from flask import signals
    app = Flask(__name__) @app.before_request
    def f6():
    g.xx = 123
    print('f6') @app.route('/index/')
    def index():
    print('index')
    return render_template('index.html') @app.route('/order')
    def order():
    print('order')
    return render_template('order.html') if __name__ == '__main__':
    app.run()
  • 视图函数

  • before_render_template / rendered_template

    # 代码示例
    
    from flask import Flask,render_template,g
    from flask import signals
    app = Flask(__name__) @signals.before_render_template.connect
    def f7(app, template, context):
    print('f7') @signals.template_rendered.connect
    def f8(app, template, context):
    print('f8') @app.route('/index/')
    def index():
    return render_template('index.html') @app.route('/order')
    def order():
    print('order')
    return render_template('order.html') if __name__ == '__main__':
    app.run()
  • after_request

    # 代码示例
    
    from flask import Flask,render_template,g
    from flask import signals
    app = Flask(__name__) @app.after_request
    def f9(response):
    print('f9')
    return response @app.route('/index/')
    def index():
    return render_template('index.html') @app.route('/order')
    def order():
    print('order')
    return render_template('order.html') if __name__ == '__main__':
    app.run()
  • request_finished

    # 代码示例
    
    from flask import Flask,render_template,g
    from flask import signals
    app = Flask(__name__) @signals.request_finished.connect
    def f10(app,response):
    print('f10') @app.route('/index/')
    def index():
    return render_template('index.html') @app.route('/order')
    def order():
    print('order')
    return render_template('order.html') if __name__ == '__main__':
    app.run()
  • got_request_exception

    # 代码示例
    
    from flask import Flask,render_template,g
    from flask import signals
    app = Flask(__name__) @app.before_first_request
    def test():
    int('asdf') @signals.got_request_exception.connect
    def f11(app,exception):
    print('f11') @app.route('/index/')
    def index():
    return render_template('index.html') @app.route('/order')
    def order():
    print('order')
    return render_template('order.html') if __name__ == '__main__':
    app.run()
  • teardown_request

    # 代码示例
    
    from flask import Flask,render_template,g
    from flask import signals
    app = Flask(__name__) @app.teardown_request
    def f12(exc):
    print('f12') @app.route('/index/')
    def index():
    return render_template('index.html') @app.route('/order')
    def order():
    print('order')
    return render_template('order.html') if __name__ == '__main__':
    app.run()
  • request_tearing_down

    # 代码示例
    
    from flask import Flask,render_template,g
    from flask import signals
    app = Flask(__name__) @signals.request_tearing_down.connect
    def f13(app,exc):
    print('f13') @app.route('/index/')
    def index():
    return render_template('index.html') @app.route('/order')
    def order():
    print('order')
    return render_template('order.html') if __name__ == '__main__':
    app.run()
  • appcontext_popped

    # 代码示例
    
    from flask import Flask,render_template,g
    from flask import signals
    app = Flask(__name__) @signals.appcontext_popped.connect
    def f14(app):
    print('f14') @app.route('/index/')
    def index():
    return render_template('index.html') @app.route('/order')
    def order():
    print('order')
    return render_template('order.html') if __name__ == '__main__':
    app.run()
  • 扩展:flash

    flash存值之后只能取一次

    # 代码示例
    
    from flask import Flask,render_template,flash,get_flashed_messages,session
    from flask import signals
    app = Flask(__name__)
    app.secret_key = 'iuknsoiuwknlskjdf' @app.route('/index/')
    def index():
    # flash('123')
    session['k1'] = 123
    return render_template('index.html') @app.route('/order')
    def order():
    # messages = get_flashed_messages()
    # print(messages)
    val = session['k1']
    del session['k1']
    print(val)
    return render_template('order.html') if __name__ == '__main__':
    app.run()

总结:

  • 关于flask内部共有14+个扩展点用于我们对flask框架内部进行定制,其中有:9个是信号。
# 源码示例
template_rendered = _signals.signal("template-rendered")
before_render_template = _signals.signal("before-render-template")
request_started = _signals.signal("request-started")
request_finished = _signals.signal("request-finished")
request_tearing_down = _signals.signal("request-tearing-down")
got_request_exception = _signals.signal("got-request-exception")
appcontext_tearing_down = _signals.signal("appcontext-tearing-down")
appcontext_pushed = _signals.signal("appcontext-pushed")
appcontext_popped = _signals.signal("appcontext-popped")
message_flashed = _signals.signal("message-flashed")
执行顺序
2. request_started = _signals.signal('request-started')                # 请求到来前执行
5. request_finished = _signals.signal('request-finished') # 请求结束后执行
3. before_render_template = _signals.signal('before-render-template') # 模板渲染前执行
4. template_rendered = _signals.signal('template-rendered') # 模板渲染后执行 发生在2/3/4/5或不执行 got_request_exception = _signals.signal('got-request-exception') # 请求执行出现异常时执行 6. request_tearing_down = _signals.signal('request-tearing-down') # 请求执行完毕后自动执行(无论成功与否)
7. appcontext_tearing_down = _signals.signal('appcontext-tearing-down')# 请求上下文执行完毕后自动执行(无论成功与否)
1. appcontext_pushed = _signals.signal('appcontext-pushed') # 请求app上下文push时执行 8. appcontext_popped = _signals.signal('appcontext-popped') # 请求上下文pop时执行 message_flashed = _signals.signal('message-flashed') # 调用flask在其中添加数据时,自动触发

自定义信号

 第一步:创建信号

第二步:将函数注册到信号中: 添加到这个列表

 第三步: 发送信号

 第四步:运行

具体实现:可参考flask源码,写一个自定义信号

from flask import Flask,flash
from flask.signals import _signals
app = Flask(__name__) xinhao = _signals.signal("xinhao")#创建信号
#定义函数
def wahaha(*args,**kwargs):
print("",args,kwargs) def sww(*args,**kwargs):
print("",args,kwargs)
# 将函数注册到信号中,添加到这个列表
xinhao.connect(wahaha)
xinhao.connect(sww) @app.route("/zzz")
def zzz():
xinhao.send(sender='xxx',a1=123,a2=456) #触发这个信号,执行注册到这个信号列表中的所有函数,此处的参数个数需要与定义的函数中的参数一致
return "发送信号成功" if __name__ == '__main__':
app.run(debug=True) #打印结果
# 111 (None,) {'sender': 'xxx', 'a1': 123, 'a2': 456}
# 222 (None,) {'sender': 'xxx', 'a1': 123, 'a2': 456}
 

flask 源码专题(九):flask扩展点的更多相关文章

  1. Flask源码解读--所有可扩展点

    一.前言 flask中有很多可扩展点(笔者这样称呼),其中包含了信号和请求钩子,这些信号和钩子有什么用呢?其主要作用用于帮助我们进行程序的耦合性,当然还可以让我们自定义一些行为.话不多说,通过阅读源码 ...

  2. flask 源码专题(五):SqlAlchemy 中操作数据库时session和scoped_session的区别

    1原生session: from sqlalchemy.orm import sessionmaker from sqlalchemy import create_engine from sqlalc ...

  3. flask 源码专题(二):请求上下文与全文上下文

    源码解析 0. 请求入口 if __name__ == '__main__': app.run() def run(self, host=None, port=None, debug=None, lo ...

  4. 08 Flask源码剖析之flask拓展点

    08 Flask源码剖析之flask拓展点 1. 信号(源码) 信号,是在flask框架中为我们预留的钩子,让我们可以进行一些自定义操作. pip3 install blinker 2. 根据flas ...

  5. flask 源码专题(十一):LocalStack和Local对象实现栈的管理

    目录 04 LocalStack和Local对象实现栈的管理 1.源码入口 1. flask源码关于local的实现 2. flask源码关于localstack的实现 3. 总结 04 LocalS ...

  6. flask 源码专题(八):路由加载

    1.示例代码 from flask import Flask app = Flask(__name__,static_url_path='/xx') @app.route('/index') def ...

  7. flask 源码专题(六):session处理机制

    前言 flask_session是flask框架实现session功能的一个插件,用来替代flask自带的session实现机制,flask默认的session信息保存在cookie中,不够安全和灵活 ...

  8. flask 源码专题(一):app.run()的背后

    当我们用Flask写好一个app后, 运行app.run()表示监听指定的端口, 对收到的request运行app生成response并返回. 现在分析一下, 运行app.run()后具体发生了什么事 ...

  9. flask 源码专题(十):flash源码研究

    flash源码 def flash(message, category="message"): flashes = session.get("_flashes" ...

随机推荐

  1. Dedecms中{dede:type}标签支持调用父级栏目名称

    需求: 我们用{dede:type}标签调用栏目相关内容时,同时需要调用该栏目的父级栏目的名称. {dede:type}标签的代码做了一下开发,支持这个调用了. 开发方法: 1.打开include/t ...

  2. yii2中的场景使用

    下面给大家介绍一下 yii2.0 场景的使用.小伙多唠叨一下了,就是担心有的人还不知道,举个简单的例子,现在在 post表里面有 title image content 三个的字段,当我创建一个 po ...

  3. [每日一题2020.06.14]leetcode #70 爬楼梯 斐波那契数列 记忆化搜索 递推通项公式

    题目链接 题意 : 求斐波那契数列第n项 很简单一道题, 写它是因为想水一篇博客 勾起了我的回忆 首先, 求斐波那契数列, 一定 不 要 用 递归 ! 依稀记得当年校赛, 我在第一题交了20发超时, ...

  4. Zookeeper——分布式一致性协议及Zookeeper Leader选举原理

    文章目录 一.引言 二.从ACID到CAP/BASE 三.分布式一致性协议 1. 2PC和3PC 2PC 发起事务请求 事务提交/回滚 3PC canCommit preCommit doCommit ...

  5. Git配置仓库的用户名邮箱

    Git配置单个仓库的用户名邮箱 $ git config user.name "gitlab's Name" $ git config user.email "gitla ...

  6. 网站搬家之mysql 5.7 date类型默认值不能设置‘0000-00-00’的问题

    网站搬家,mysql版本由5.6升级到5.7,遇到问题: mysql 5.7之后版本datetime默认值设置'0000-00-00',出现异常:Invalid default value for ' ...

  7. 黎活明8天快速掌握android视频教程--23_网络通信之网络图片查看器

    1.首先新建立一个java web项目的工程.使用的是myeclipe开发软件 图片的下载路径是http://192.168.1.103:8080/lihuoming_23/3.png 当前手机和电脑 ...

  8. Python3-random模块-生成伪随机数

    Python3中的random模块可以生成一个伪随机数 常用的就三个模块函数 random.random() 返回0-1之间的一个随机数 random.randrange(i, j) 返回一个随机数 ...

  9. SpringBoot--使用redis实现分布式限流

    1.引入依赖 <!-- 默认就内嵌了Tomcat 容器,如需要更换容器也极其简单--> <dependency> <groupId>org.springframew ...

  10. js基础练习题(2)

    5.函数 1.按要求封装两个函数 1.封装一个函数,要求输入字符串转化成数组弹出 2.封装一个函数,要求能求出三个数中的最小值,注意:不准使用js内置函数 2.封装一个函数,求参数的和,注意:参数不固 ...