Flask 第十五话之请求上下文及全局全局存储g对象

一、简介

在 flask 中,视图函数需要知道它执行情况的请求信息(请求的 url,参数,方法等)以及应用信息(应用中初始化的数据库等),才能够正确运行。

最直观地做法是把这些信息封装成一个对象,作为参数传递给视图函数。但是这样的话,所有的视图函数都需要添加对应的参数,即使该函数内部并没有使用到它。

flask 的做法是把这些信息作为类似全局变量的东西,视图函数需要的时候,可以使用 from flask import request 获取。但是这些对象和全局变量不同的是——它们必须是动态的,因为在多线程或者多协程的情况下,每个线程或者协程获取的都是自己独特的对象,不会互相干扰。

那么如何实现这种效果呢?如果对 python 多线程比较熟悉的话,应该知道多线程中有个非常类似的概念  threading.local,可以实现多线程访问某个变量的时候只看到自己的数据。内部的原理说起来也很简单,这个对象有一个字典,保存了线程 id 对应的数据,读取该对象的时候,它动态地查询当前线程 id 对应的数据。

flask 中有两种上下文:application context 和 request context。上下文有关的内容定义在 globals.py 文件。

二、应用上下文使用

from flask import Flask,request,session,current_app,url_for,g
from werkzeug.local import Local,LocalStack

app = Flask(__name__)

# 应用上下文:写法一
app_context  = app.app_context()
app_context.push()
print(current_app.name)

# 应用上下文:写法二
# 只能在with函数内容
with app.app_context():
    print(current_app.name)

@app.route('/')
def hello_world():
    print(url_for('mylist'))
    return 'Hello World!'


if __name__ == '__main__':
    app.run()

三、请求上下文使用

from flask import Flask,request,session,current_app,url_for,g
from werkzeug.local import Local,LocalStack

app = Flask(__name__)

@app.route('/')
def hello_world():
    print(url_for('mylist'))
    return 'Hello World!'


@app.route('/list/')
def mylist():
    return 'Hello World!'

# 请求上下文
with app.test_request_context():
    # 手动推入一个请求上下文到请求上下文栈中
    # 如果没有请求上下文,那么先推入一个应用上下文到栈中
    print(url_for('mylist'))

if __name__ == '__main__':
    app.run()

 四、g对象

注:g对象是全局flask中运行时都可以访问的对象,并且和request一样也是线程隔离的,是专门用于开发者自己定义的存储对象,方便于在整个项目组获取和使用

from flask import Flask,request,g
from utils import *

app = Flask(__name__)

@app.route('/')
def hello_world():
    username = request.args.get('username')
    g.username = username
    log_a()
    log_b()
    log_c()
    print(url_for('mylist'))
    return 'Hello World!'


if __name__ == '__main__':
    app.run()

utils.py:其他函数中调用g对象

from flask import g

def log_a():
    print('log a %s'%g.username)

def log_b():
    print('log b %s'%g.username)

def log_c():
    print('log c %s'%g.username)

 

posted @ 2020-03-15 11:02  我在地球凑人数的日子  阅读(688)  评论(0编辑  收藏  举报