second_请求钩子_异常捕获_上下文及g变量_flask-script

2. 请求钩子
在客户端和服务器交互的过程中,有些准备工作或扫尾工作需要处理,比如:
-
在请求开始时,建立数据库连接;
-
在请求开始时,根据需求进行权限校验;
-
在请求结束时,指定数据的交互格式;
为了让每个视图函数避免编写重复功能的代码,Flask提供了通用设置的功能,即请求钩子。
请求钩子是通过装饰器的形式实现,Flask支持如下四种请求钩子:
-
before_first_request
-
在处理第一个请求前执行
-
-
before_request
-
在每次请求前执行
-
如果在某修饰的函数中返回了一个响应,视图函数将不再被调用
-
-
after_request
-
如果没有抛出错误,在每次请求后执行
-
接受一个参数:视图函数作出的响应
-
在此函数中可以对响应值在返回之前做最后一步修改处理
-
需要将参数中的响应在此参数中进行返回
-
-
teardown_request:
-
在每次请求后执行
-
接受一个参数:错误信息,如果有相关错误抛出
-
-
from flask import Flask from settings.dev import DevConfig app = Flask(__name__) # 项目配置 app.config.from_object(DevConfig) @app.before_first_request def before_first_request(): print("----before_first_request----") print("系统初始化的时候,执行这个钩子方法") print("会在接收到第一个客户端请求时,执行这里的代码") @app.before_request def before_request(): print("----before_request----") print("每一次接收到客户端请求时,执行这个钩子方法") print("一般可以用来判断权限,或者转换路由参数或者预处理客户端请求的数据") @app.after_request def after_request(response): print("----after_request----") print("在处理请求以后,执行这个钩子方法") print("一般可以用于记录会员/管理员的操作历史,浏览历史,清理收尾的工作") response.headers["Content-Type"] = "application/json" # 必须返回response参数 return response class APIError(Exception): print("-----111111111") @app.teardown_request def teardown_request(exc): print("----teardown_request----") print("在每一次请求以后,执行这个钩子方法,如果有异常错误,则会传递错误异常对象到当前方法的参数中") print(exc) @app.route("/") def index(): print("----视图函数----") raise APIError("报错了") print("视图函数被运行了") return "视图函数被运行了<br>" if __name__ == '__main__': app.run() # app.run(host="0.0.0.0", port=80)
3. 异常捕获
主动抛出HTTP异常
-
-
抛出一个给定状态代码的 HTTPException 或者 指定响应,例如想要用一个页面未找到异常来终止请求,你可以调用 abort(404)。
-
-
参数:
-
code – HTTP的错误状态码
-
# abort(404) 抛出状态码的话,只能抛出 HTTP 协议的错误状态码 abort(500)
-
-
注册一个错误处理程序,当程序抛出指定错误状态码的时候,就会调用该装饰器所装饰的方法
-
-
参数:
-
code_or_exception – HTTP的错误状态码或指定异常
-
-
例如统一处理状态码为500的错误给用户友好的提示:
@app.errorhandler(500) def internal_server_error(e): return '服务器搬家了'
捕获指定异常类型
@app.errorhandler(ZeroDivisionError) def zero_division_error(e): return '除数不能为0'
4. 上下文及g变量
Flask中有两种上下文,请求上下文(request context)和应用上下文(application context)。
Flask中上下文对象:相当于一个容器,保存了 Flask 程序运行过程中的一些信息。
-
application 指的就是当你调用
app = Flask(__name__)创建的这个对象app; -
request 指的是每次
http请求发生时,WSGI server(比如gunicorn)调用Flask.__call__()之后,在Flask对象内部创建的Request对象; -
application 表示用于响应WSGI请求的应用本身,request 表示每次http请求;
-
application的生命周期大于request,一个application存活期间,可能发生多次http请求,所以,也就会有多个request
请求上下文(request context)
思考:在视图函数中,如何取到当前请求的相关数据?比如:请求地址,请求方式,cookie等等
在 flask 中,可以直接在视图函数中使用 request 这个对象进行获取相关数据,而 request 就是请求上下文的对象,保存了当前本次请求的相关数据,请求上下文对象有:request、session
-
request
-
-
session
-
用来记录请求会话中的信息,针对的是用户信息。举例:session['name'] = user.id,可以记录用户信息。还可以通过session.get('name')获取用户信息。
-
应用上下文(application context)
它的字面意思是 应用上下文,但它不是一直存在的,它只是request context 中的一个对 app 的代理(人),所谓local proxy。它的作用主要是帮助 request 获取当前的应用,它是伴 request 而生,随 request 而灭的。
应用上下文对象有:current_app,g
current_app
应用程序上下文,用于存储应用程序中的变量,可以通过current_app.name打印当前app的名称,也可以在current_app中存储一些变量,例如:
-
应用的启动脚本是哪个文件,启动时指定了哪些参数
-
加载了哪些配置文件,导入了哪些配置
-
连接了哪个数据库
-
有哪些可以调用的工具类、常量
-
current_app.name current_app.test_value='value'
from flask import Flask from config import Config from flask import request app = Flask(__name__) app.config.from_object(Config) @app.route('/') def index(): return "the views was run" from flask import g @app.route("/context2") class Model2(object): def __init__(self): print("model receive data, num = %s" % g.username) from flask import current_app @app.route("/context3") def context3(): print(current_app.username) return "apply the before and next contents" @app.route("/context4") def context4(): g.username = request.args.get("username") Model2() return "应用上下文4" if __name__ == '__main__': app.username = "apply the before and next contents" app.run()
此时测试访问/context3或/context4
g变量
g.name='abc' # 注意:不同的请求,会有不同的全局变量
两者区别:
-
请求上下文:保存了客户端和服务器交互的数据
-
5.
# 安装命令: pip install flask-script
# main.py
from flask import Flask from config import Config from flask import request app = Flask(__name__) app.config.from_object(Config) from flask_script import Manager manage = Manager(app) from flask_script import Command class HelloCommand(Command): def run(self): with open("text.txt", "w")as f: f.write("hello\r\nhello") pass print("this is run the order of hello") manage.add_command("hello", HelloCommand()) @app.route('/') def index(): return "the views was run" if __name__ == '__main__': manage.run()
# manage.py
import sys if __name__ == "__main__": print(sys.argv)
集成 Flask-Script到flask应用中

Flask-Script 可以为当前应用程序添加脚本命令

Flask内置的模板语言,它的设计思想来源于 Django 的模板引擎,并扩展了其语法和一系列强大的功能。
渲染模版函数
-
Flask提供的 render_template 函数封装了该模板引擎
-
render_template 函数的第一个参数是模板的文件名,后面的参数都是键值对,表示模板中变量对应的真实值。
1. 在视图函数中设置渲染模板
2.
Jinja2 模版中的变量代码块可以是任意 Python 类型或者对象,只要它能够被 Python 的 str() 方法转换为一个字符串就可以,比如,可以通过下面的方式显示一个字典或者列表中的某个元素:
模板中特有的变量和函数
| 描述 | |
|---|---|
loop.index |
当前循环迭代的次数(从 1 开始) |
loop.index0 |
当前循环迭代的次数(从 0 开始) |
loop.revindex |
到循环结束需要迭代的次数(从 1 开始) |
loop.revindex0 |
到循环结束需要迭代的次数(从 0 开始) |
loop.first |
如果是第一次迭代,为 True 。 |
loop.last |
如果是最后一次迭代,为 True 。 |
loop.length |
序列中的项目数。 |
loop.cycle |
在一串序列间期取值的辅助函数。见下面示例程序。 |
-
比如:要是我们想知道当前被迭代的元素序号,并模拟Python中的enumerate函数做的事情,则可以使用loop变量的index属性,例如:
过滤器
自定义过滤器
过滤器的本质是函数。当模板内置的过滤器不能满足需求,可以自定义过滤器。自定义过滤器有两种实现方式:
-
一种是通过Flask应用对象的 add_template_filter 方法
-
通过装饰器来实现自定义过滤器
重要:自定义的过滤器名称如果和内置的过滤器重名,会覆盖内置的过滤器。
# main.py from flask import Flask from config import Config from flask import request app = Flask(__name__, template_folder="templates") # app = Flask(__name__) app.config.from_object(Config) # g过滤器的注册及使用 from flask import g # 渲染模块的调用 from flask import render_template @app.route('/') def index(): title = "网页标题" dict1 = {"id": 1, "username": "xiaoming", "money": 10.5} g.list1 = ["睡觉", "吹牛", "敲代码"] g.book_list = [ {"id": 11, "name": "浪潮之巅", "price": 88.5}, {"id": 22, "name": "数学之美", "price": 68.5}, {"id": 33, "name": "流畅的Python", "price": 118.5}, {"id": 44, "name": "五年高考三年模拟", "price": 78.5}, ] # 返回渲染模块 return render_template("index.html", title=title, dict1=dict1) # 自定义过滤器 def rev(data): data.reverse() return data # 注册过滤器 app.add_template_filter(rev, "myrev") @app.route("/filter") def myfileter(): g.list1 = ["睡觉", "吹牛", "敲代码"] return render_template("f.html") if __name__ == '__main__': app.run()
# templates/f.html <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> {#自带过滤器的使用#} {{ "hello world" | reverse | upper }} {{ g.list1 }}<br> {#自定义过滤器的使用#} {{ g.list1 |myrev }} </body> </html>
# templates/index.html <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>{{ title }}</title> </head> <body> <h1>{{ title }}</h1> {#g 本次全局变量的使用#} <p>{{ g.list1 }}</p> <table> <tr> <th>order</th> <th>ID</th> <th>name</th> <th>price</th> </tr> {#判断语句的使用#} {% for book in g.book_list %} {% if loop.first %} <tr style="color: red"> {% elif loop.last %} <tr style="color: blue"> {% else %} <tr> {% endif %} <td>{{ loop.index }}</td> <td>{{ book.id }}</td> <td>{{ book.name }}</td> <td>{{ book.price }}</td> </tr> {% endfor %} </table> {#模板中特有的变量和函数#} {{ request.url }} {{ session.new }} {{ g.name }} </body> </html>

浙公网安备 33010602011771号