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. GET与POST的区别

    转自http://blog.csdn.net/darxin/article/details/4944225#comments HTTP请求的GET与POST方式的本质区别可以参考hyddd在<浅 ...

  2. mysql 忘记root 密码的解决方法

    LINUX 1.切换root 用户 2.停止mysqld 服务 /etc/inid.d mysqld stop 3.跳过验证登录 mysqld_safe --skip-grant-tables &am ...

  3. noSuchMethodException问题

    上午遇到一个nosuchMethodException 折腾了一上午发现是jar包冲突引起的.首先发现单独运行没问题,和其他项目整合后就有问题,当时以为代码问题,其实早该想到是jar包冲突造成的... ...

  4. Windows2008 R2 Enterprise离线安装IE10和VS2015过程记录

    直接下载IE10,进行安装,提示需要联机下载更新: 在网上搜索到一篇文章(http://www.cnblogs.com/nbpowerboy/p/3383992.html),参考 以下载简体中文的Wi ...

  5. 我读汤姆大叔的深入理解js(一)

    前言 闲来看看javascript,在圆子里发现了汤姆大叔的文章,先是整体瞄了几眼,感觉不错,然后细细研读.记录下自己的学习历程和个人理解.更重要的是作为笔记 高质量JS代码 在看汤姆大叔的这一系列文 ...

  6. poj1068

    Parencodings Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 18785   Accepted: 11320 De ...

  7. Java 分页通用

    1.定义分页模型:PageModel package com.common.page; import java.util.List; /** * 封装分页信息 * @author Administra ...

  8. ACM之路(20)—— Splay初探

    由于数据结构上老师讲了AVL树的rotate,然后去学了一下treap和Splay,这些数据结构还真是神奇啊! treap暂时只知道名次树的作用(就是一段动态变化的有序数列,找第K大的元素,用set显 ...

  9. 如何在HTML中加载Flash(2种实现方法)_HTML/Xhtml_网页制作

    点评:如何在HTML中加载Flash,为网页添加更多的色彩,普通的网页以无法满足用户的需求,接下来为大家介绍下2种在HTML中加载Flash的方法,感兴趣的各位可以适当参考下,希望对你有所帮助 第一种 ...

  10. Altium Designer XX 重新定义板框形状和大小的方法

    Altium Designer15 重新定义板框形状和大小的方法:重新定义板框形状和大小的方法.很简单,点击数字键"1",就会看到板框界面变绿了这时候你在去点击菜单栏里的Desig ...