首页 > 代码库 > 2.5.1、程序和请求上下文

2.5.1、程序和请求上下文

    Flask 从客户端收到请求时,要让视图函数能访问一些对象,这样才能处理请求。请求对象就是一个很好的例子,它封装了客户端发送的 HTTP 请求。

    要想让视图函数能够访问请求对象,一个显而易见的方式是将其作为参数传入视图函数,不过这会导致程序中的每个视图函数都增加一个参数。

    除了访问请求对象,如果视图函数在处理请求时还要访问其他对象,情况会变得更糟。

    为了避免大量可有可无的参数把视图函数弄得一团糟,Flask 使用上下文临时把某些对象变为全局可访问。有了上下文,就可以写出下面的视图函数:

from flask import request

 

@app.route(‘/‘)

def index():

    user_agent = request.headers.get(‘User-Agent‘)

    return ‘<p>Your browser is %s</p>‘ % user_agent

Your b rowser is use r_agent app. rou e user def use r(name): 0/0 name http://127.0.0.l:sooo/ -"hi-1-: GET ? ‘00 0K HTTP/I O Date: ‘Wed, 29 Mar 2017 GMT‘ Server: "We.-kzeug/O.12.1 Python/2.7. 5‘ - 10.332 KB) User—Agent: "Mozilla/5.O (XII; Linux x86 _ 64; if retu rn name 52.0) Gecko/20100101 Firefox/S2.O‘ maln app. run(debug-T rue) ccept: 9, Accept—Encoding: "gzip, deflate‘ Connection: "keep-alive Cache—Control: "max-age—O " height="450" width="1097">

注意在这个视图函数中我们如何把 request 当作全局变量使用。事实上,request 不可能是

全局变量。试想,在多线程服务器中,多个线程同时处理不同客户端发送的不同请求时,

每个线程看到的 request 对象必然不同。Falsk 使用上下文让特定的变量在一个线程中全局

可访问,与此同时却不会干扰其他线程。

 

注:线程是可单独管理的最小指令集。进程经常使用多个活动线程,有时还会共

享内存或文件句柄等资源。多线程 Web 服务器会创建一个线程池,再从线

程池中选择一个线程用于处理接收到的请求。

 

在 Flask 中有两种上下文:程序上下文和请求上下文。表 2-1 列出了这两种上下文提供的

变量。

表2-1 Flask上下文全局变量

 技术分享

 

Flask 在分发请求之前激活(或推送)程序和请求上下文,请求处理完成后再将其删除。程

序上下文被推送后,就可以在线程中使用 current_app 和 g 变量。类似地,请求上下文被

推送后,就可以使用 request 和 session 变量。如果使用这些变量时我们没有激活程序上

下文或请求上下文,就会导致错误。如果你不知道为什么这 4 个上下文变量如此有用,先

别担心,后面的章节会详细说明。

下面这个 Python shell 会话演示了程序上下文的使用方法:

[user@localhost test]$ source venv/bin/activate

(venv)[user@localhost test]$ python

Python 2.7.5 (default, Nov  6 2016, 00:28:07)

[GCC 4.8.5 20150623 (Red Hat 4.8.5-11)] on linux2

Type "help", "copyright", "credits" or "license" for more information.

>>> from hello import app

>>> from flask import current_app

>>> current_app.name

Traceback (most recent call last):

...

RuntimeError: working outside of application context

>>> app_ctx = app.app_context()

>>> app_ctx.push()

>>> current_app

<Flask ‘hello‘>

>>> app_ctx.pop()

>>>

在这个例子中,没激活程序上下文之前就调用 current_app.name 会导致错误,但推送完上下文之后就可以调用了。注意,在程序实例上调用 app.app_context() 可获得一个程序上

下文。

2.5.1、程序和请求上下文