Flask备注三(Context)

Flask支持不同的应用场景下,对应不同的local context(本地上下文环境),用来提供当前环境下的资源。lcoal context和全局变量以及局部变量最大的不同在于,作用域是代码范围的,而local context是应用场景范围的。

Flask支持的local context包含application context以及request context。这两个都是线程安全的(基于当前的线程和request), application context存储的是和当前应用程序相关的资源,request context存储的是和当前request相关的资源和信息。

proxies

理解local context之前,首先需要理解flask的proxies。如之前所述,Flask的接口提供了一些常用的全局变量:

  • flask.current_app: 当前线程所在的application,flask支持在一个python进程中运行多个app,因此可以通过current_app获取当前所在app。
  • flask.request: 当前处理的request,只有在request处理状态可用。
  • flask.g: 用来保存状态和资源信息等任何信息。
  • flask.session: 当前app的session信息。

上述全局变量都是proxy实现的即:使用这些proxy会访问proxy指向的合适的真实对象。通过这种proxy机制,不同线程使用proxy变量时,代理根据当前的应用场景指向对应线程的变量。因此确保了变量的线程安全,并对最终对象完成了封装(大多数情况下,使用proxy,不需要关心最终对象)。但是下述情况下需要获取proxy对应的最终对象:

  • proxy并不会模拟最终对象的类型信息,如果要做示例的类型判断,则需要获取最终对象。
  • flask的singals需要传递当前application的真是引用,因此要获取最终对象。获取示例如下:

    app = current_app._get_current_object()
    my_signal.send(app)

proxy指向的对象是线程安全的,并且在不同的应用场景下不同,因此一般储存在application context 和 request context中。

request context

request conetext提供了在request处理时的上线文环境,提供了和request相关的资源。request context提供的资源只能在request处理的线程中使用。在服务器获取request时创建,并在response返回或者exception处理后销毁。

在Flask中维护一个_request_ctx_stack存储所有的request context并维持request context的生命周期。在Flask的通用request处理流程中,request context的实现如下:

  1. 通过request_context函数创建request context的对象。

    test_request_contet函数提供用户代码模拟创建request context。

  2. request context的push方法,将request context存储到_request_ctx_stack中。

  3. 当前request进行处理,将需要的资源存储到request context中。
  4. 在request处理完成或者exception处理完成后,request context通过pop方法将request context从_request_ctx_stack中弹出并释放。

    reqeust context的生命周期和request的依赖于request的处理周期,在Flask中request的处理周期如下:

    1. 在request被处理之前,所有的before_request的handler执行,如果其中有response返回则后续功能不再执行,response会替换预期的view的response。
    2. 如果没有before_request的handler返回response,则开始正常的request处理并将预定的view作为response。
    3. 正常response传入afterrequest的handler,在afterrequst的handler中可以对response进行修改和替换。
    4. 最终在request处理结束时执行teardownrequest的Handler。teardownrequest handler永远会被执行,无论是否出错以及是否执行before_request handler。

程序代码以及第三方扩展都可以修改request context保存需要的资源(更好的方式是将额外的资源存在在application context中)。reqeust context默认提供的资源包含:

  • reqeust: 当前的request对象。
  • url_adapter: 当前request所使用的url adapter。
  • session: 当前的session对象。
  • flashes: flash消息的缓存。

application context

application context提供了和当前运行application相关的local context,同样是只在当前request和线程中可用,因此也是线程安全的。生命周期和request context类似,在创建requst context时如果当前线程不存在application context,则同时创建application context。

和request context类似在application context中维护了一个_app_ctx_stack存储所有的app context并维持application context的声明周期。和request conext不同,application context默认提供的资源包含:

  • app: 当前线程所在的application,可以通过flask.current_app proxy访问。
  • g : 存储用户定义以及第三方扩展需要的资源及属性。

相比于request context,更推荐在application context中存储用户定义以及第三方扩展需要的资源。在local context中存储用户自定义资源必须要明确两点:

  • 在context中,进行隐式的存储。(在aplication context中,存储到flask.g中)
  • 在context的销毁时,完成资源的销毁。

在local context的自定义的resource,一般通过get_X()函数创建,通过teardown_X()函数销毁,其中teardown_X()函数必须注册为application context或者request context的handler。示例如下:

import sqlite3
from flask import g def get_db():
db = getattr(g, '_database', None)
if db is None:
db = g._database = connect_to_database()
return db @app.teardown_appcontext
def teardown_db(exception):
db = getattr(g, '_database', None)
if db is not None:
db.close()

同样,和其他的proxy一样,可以将这个资源声明为LocalProxy。

from werkzeug.local import LocalProxy
db = LocalProxy(get_db)

这样我们可以通过db来访问get_db函数。

Flask备注三(Context)的更多相关文章

  1. Flask 备注一(单元测试,Debugger, Logger)

    Flask 备注一(单元测试,Debugger, Logger) Flask是一个使用python开发Web程序的框架.依赖于Werkzeug提供完整的WSGI支持,以及Jinja2提供templat ...

  2. Flask备注二(Configurations, Signals)

    Flask备注二(Configuration, Signals) Flask是一个使用python开发Web程序的框架.依赖于Werkzeug提供完整的WSGI支持,以及Jinja2提供templat ...

  3. Flask备注4(Structure)

    Flask备注4(Structure) package 通过Flask可以非常简单的通过一个module(一个py文件)创建一个简单的application.这种简单程序的文件结构如下: /youra ...

  4. Flask 上下文(Context)原理解析

    :first-child { margin-top: 0; } blockquote > :last-child { margin-bottom: 0; } img { border: 0; m ...

  5. flask基础三

    一.路由和视图(基础二上补充) (1)有参装饰器 路由采用的是有参装饰器实现的 @app.route("/index",methods=["GET"," ...

  6. docker 部署 flask(三)高级编写及生成镜像,安装requirements.txt

    简介: 上一篇,我写了如何使用别人的docker基础镜像,生成我们的docker镜像. 也就最基本的flask,没有别的库(包)支持.连数据库支持都没有. 也就让大家了解一下怎么生成镜像而已. 本篇介 ...

  7. Flask第三篇——安装Flask

    现在我们来安装Flask: Windows系统安装Flask 如果你的系统是Windows,那安装起来非常方便——pip install flask Mac系统安装Flask Mac安装Flask一般 ...

  8. flask系列三之Jinja2模板

    1.如何渲染模板 模板在‘templates’文件夹下(htnl页面) 从flask中导入render_template函数---渲染html模板 在视图函数中,使用render_template 函 ...

  9. flask 学习(三)

    继续flask的学习.尝试了使用程序context这一部分: 而在hello.py文档的旁边发现新出现了hello.pyc,看来运行过程中也被编译成字节码文件了,也不清楚是在哪个步骤的,留着后面研究. ...

随机推荐

  1. Vuejs使用笔记 --- component内部实现

    现在来系统地学习一下Vue(参考vue.js官方文档): Vue.js是一个构建数据驱动的web界面的库,其目标是实现响应的数据绑定和组合的试图组件. Vue.js拥抱数据驱动的视图概念,这意味着我们 ...

  2. Webservice接口

    快递查询接口 http://webservice.36wu.com/ExpressService.asmxip查询接口 http://webservice.36wu.com/ipService.asm ...

  3. shell 脚本,提取文件中的内容

    使用awk.cut.sed.if.while 等 awk.cut.sed还是很重要的 这是后来修改的,可以完成 #!/bin/bash #conver formatFILE=mobile_dpi.ru ...

  4. 答:SQLServer DBA 三十问之三:有哪些操作会使用到TempDB;如果TempDB异常变大,可能的原因是什么,该如何处理

    3. 有哪些操作会使用到TempDB:如果TempDB异常变大,可能的原因是什么,该如何处理: tempdb的用途:1)存储专用和全局临时变量,不考虑数据库上下文: 2)与Order by 子句,游标 ...

  5. 记录js学习之this用法

    一直对Javascript中的this都有一种似是而非的感觉,今天突然感觉豁然开朗,特此记录一下. 咱们先看个栗子:    <!DOCTYPE html><html><h ...

  6. 泛型(Generic)

    本质:限制集合类型 我们在编写泛化类的时候,我们要时刻提醒自己,我们传入的参数T仅仅是一个Object类型,任何具体类型信息我们都是未知的. 小例子: package day02.generic; i ...

  7. WeedFS问题收集

    weed fs FAQ Moving files to a different server Q: Hello,is it possible to move manually the .dat and ...

  8. Vim配置(k-vim)

    无意发现一个vim的插件 转载:http://www.cnblogs.com/yxy2829/p/5250587.html 截图 solarized主题 molokai主题 安装步骤 1. clone ...

  9. Redis_DataType

    Redis_DataType.html :first-child{margin-top:0!important}img.plugin{box-shadow:0 1px 3px rgba(0,0,0,. ...

  10. SOME:收缩数据库日志文件,查看表数据量和空间占用,查看表结构索引修改时间

    ---收缩数据库日志文件 USE [master]ALTER DATABASE yourdatabasename SET RECOVERY SIMPLE WITH NO_WAITALTER DATAB ...