前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >小白学Flask第八天| Flask上下文和请求钩子

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

作者头像
Python进击者
发布2019-09-17 15:43:53
6060
发布2019-09-17 15:43:53
举报
文章被收录于专栏:JAVAandPython君

本文内容:

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)

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


本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2019-09-10,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Python进击者 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
容器服务
腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档