小白学Flask第八天| Flask上下文和请求钩子

2019-09-17 15:43:53 浏览数 (1)

本文内容:

1.Flask的上下文对象

2.请求钩子

Flask的上下文对象

在这篇文章之前,我们学习过request和session这两个小家伙,他们两的功能都非常的强大,我们今天讲上下文对象,也是和他们两有着很大的关系。

request:封装了HTTP请求的内容,针对的是http请求。举例:user = request.args.get('user'),获取的是get请求的参数。

session:用来记录请求会话中的信息,针对的是用户信息。举例:session['name'] = user.id,可以记录用户信息。还可以通过session.get('name')获取用户信息。

首先给大家看段代码:

代码语言:javascript复制
from flask import Flask, request
 
app = Flask(__name__)
 
@app.route('/')
def hello_world(request): #  在这里将request对象作为参数传进来
    data = request.json
    return 'hello world'
 
if __name__ == '__main__':
    app.run()

大家仔细看这段代码会发现我们将request对象作为参数传进视图函数hello_world,如果我们当真要这样去使用request对象,视图函数一多,我们就会发现这样非常的不整洁而且还容易出错。

为了解决这个问题,利用“上下文对象”将request对象作为全局变量,此时这个request对象就是在这个线程中的全局变量。但是如果这个对象是在A线程当中那么他就是A线程中的全局变量,在其他线程(B线程,C线程...)当中不是全局变量,这样就可以保证对象不会混淆。

所以我们平时这样写就OK了:

代码语言:javascript复制
from flask import Flask, request
 
app = Flask(__name__)
 
@app.route('/')
def hello_world():
    data = request.json
    return 'hello world'
 
if __name__ == '__main__':
    app.run()

session的道理也是类似的,这里就不多阐述。

除了request和session这类请求上下文对象(request context),还有一类上下文对象,叫做应用上下文对象(application context)

current_app和g都属于应用上下文对象。

current_app : 表示当前运行程序文件的程序实例。

g : 处理请求时,用于临时存储的对象,每次请求都会重设这个变量。

current_app在之前的文章中我们也简单介绍过。那么g是啥?

其实它就是一个存储容器,你想往里面存储什么样的数据都可以。

代码语言:javascript复制
from flask import Flask,g

app = Flask(__name__)
 
@app.route('/')
def hello_world():
    g.username = "JavaandPython君"
    g.pass = "123"
    return 'hello world'
 
if __name__ == '__main__':
    app.run()

请求钩子

大家可能以前没听说过这个概念,其实非常容易理解,大家都知道钩子是什么,钩子有什么用呢? 钩住某个东西然后跟它连在一起。

在客户端和服务器交互的过程中,有些准备工作或扫尾工作需要处理,比如:在请求开始时,建立数据库连接;在请求结束时,指定数据的交互格式。为了让每个视图函数避免编写重复功能的代码,Flask提供了通用设施的功能,即请求钩子。

这里给出几个钩子的概念:

1.before_first_request : 在第一次请求之前运行,只需执行一次,如链接数据库

2.before_request : 在每一次请求都会执行,可以在这里做权限校验操作,比如说某用户是黑名单用户,黑名单用户登录系统将遭到拒绝访问,可以使用before_request进行权限校验。

3.after_request :在请求之后运行,会接收一个参数,这个参数就是前面的请求处理完毕之后, 返回的响应数据,如果需要对响应做额外处理,可以再这里进行。

4.teardown_request :每一次请求之后都会调用,会接受一个参数,参数是服务器出现的错误信息

如何使用它们呢?其实也是非常简单

代码语言:javascript复制
from flask import Flask

app = Flask(__name__)

@app.route('/')
def helloworld():
    return 'hello world'

# 在第一次请求之前运行.
# 例子: 比如连接数据库操作, 只需要执行一次
@app.before_first_request
def before_first_request():
    print('before_first_request')

# 在每一次请求都会执行
# 例子: 可以在这里做权限校验操作,比如说某用户是黑名单用户,黑名单用户登录系统将遭到拒绝访问,可以使用
# before_request进行权限校验
@app.before_request
def before_request():
    print('before_request')

# 在请求之后运行
@app.after_request
def after_request(response):
    # response: 就是前面的请求处理完毕之后, 返回的响应数据
    # 如果需要对响应做额外处理,可以再这里进行
    # json.dumps 配置请求钩子
    # response.headers["Content-Type"] = "application/json"
    print('after_request')
    return response

# 每一次请求之后都会调用,会接受一个参数,参数是服务器出现的错误信息
@app.teardown_request
def teardown_request(error):
    # 数据库的扩展, 可以实现自动提交数据库
    print('teardown_request: error %s' % error)


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

关于钩子的知识点,我们只需要去记住这几个点就OK了。

但是这里给大家延申一个知识,从上面大家可能会发现我们这些钩子不能够去锁定某个视图函数,例如他不能确定我只有运行A视图函数才去执行钩子里的内容,我运行B视图函数他也会去执行,那么怎样才能够指定视图函数执行指定的钩子内容。

代码语言:javascript复制
from flask import Flask,url_for, request

app = Flask(__name__)

@app.route('/hello')
def helloworld():
    return 'hello world'

@app.route('/index')
def index():
    return 'hello index'
    
# 在第一次请求之前运行.
# 例子: 比如连接数据库操作, 只需要执行一次
@app.before_first_request
def before_first_request():
     path = request.path
     if path == url_for("hello"):
        printf("如果是hello视图函数就执行这个")
     elif path == url_for("index"):
        printf("如果是index视图函数就执行这个")
     return ""

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

通过上面代码,相信你就已经懂怎么做了!


0 人点赞