add_url_rule和app.route原理剖析
add_url_rule
add_url_rule(rule,endpoint=None,view_func=None)
这个方法用来添加url与视图函数的映射。
如果没有填写endpoint,那么默认会使用view_func的名字作为endpoint,
有填写endpoint,使用endpoint指定的字符串作为view_func函数的别名。
在url_for调用时应使用endpoint传递的别名
@app.route('/',endpoint='index') def hello_world(): # return 'Hello World!' + url_for('my_list') return 'Hello World!'+url_for('alias') def my_list(): return '我是列表页' app.add_url_rule('/list/',endpoint='alias',view_func=my_list) # app.test_request_context # 请求上下文 with app.test_request_context(): print(url_for('index')) #结果是:/ if __name__ == '__main__': app.run(debug=True)
add_url_rule相当于把/list/和my_list关联起来。endpioint相等于给视图函数命名一个别名。
def my_list(): return '我是列表页' app.add_url_rule('/list/',endpoint='alias',view_func=my_list) #和下面的等价: @app.route('/list/',endpoint='alias') def my_list(): return '我是列表页'
app.route原理
@app.route('/')装饰器底层是使用add_url_rule来实现url与视图函数映射的,
标准类视图及其使用场景
标准视图继承自flask.views.View,并且在子类中必须实现dispatch_request方法,这个方法类似于视图函数,也要返回一个基于Response或者其子类的对象。
- 标准类视图,必须继承自flask.views.View.
- 必须实现dipatch_request方法,当请求过来后会执行该方法且方法的返回值和之前的函数视图返回值一样,必须返回Response或者子类的对象,或者是字符串,或者是元组。
- 必须通过app.add_url_rule(rule,endpoint,view_func)来做url与视图的映射。view_func这个参数,需要使用类视图下的as_view类方法类转换:ListView.as_view('list')。
- 如果指定了endpoint,那么在使用url_for反转的时候就必须使用endpoint指定的别名。如果没有指定endpoint,那么就可以使用as_view(视图名字)中指定的视图名字来作为反转。
- 类视图有以下好处:可以继承,把一些共性的东西抽取出来放到父视图中,子视图直接拿来用。但是也不是说所有的视图都要使用类视图,这个要根据情况而定。
# 有几个url需要返回json数据 # 有几个视图,是需要返回相同的变量 class JSONView(views.View): def get_data(self): raise NotImplementedError def dispatch_request(self): return jsonify(self.get_data()) class ListView(JSONView): def get_data(self): return {"username":'wqbin','password':'123456'} app.add_url_rule('/list/',endpoint='my_list',view_func=ListView.as_view("list"))
class ADSView(views.View): def __init__(self): super(ADSView, self).__init__() self.context = { 'ads': '这里是wqbin的博客' } class LoginView(ADSView): def dispatch_request(self): self.context.update({ 'username': 'wqbin' }) return render_template('login.html',**self.context) class RegistView(ADSView): def dispatch_request(self): return render_template('regist.html',**self.context) app.add_url_rule('/login/',view_func=LoginView.as_view('login')) app.add_url_rule('/regist/',view_func=RegistView.as_view('regist'))
基于调度方法的类视图
基于请求方法的类视图:
基于方法的类视图,是根据请求的method来执行不同的方法的。
如果用户是发送的get请求,那么将会执行这个类的get方法。如果用户发送的是post请求,那么将会执行这个类的post方法。其他的method类似,比如delete、put。
这种方式,可以让代码更加简洁。
所有和get请求相关的代码都放在get方法中,所有和post请求相关的代码都放在post方法中。就不需要跟之前的函数一样,通过request.method == 'GET'。
class LoginView(views.MethodView): def __render(self,error=None): return render_template('login.html',error=error) def get(self): return self.__render() def post(self): username = request.form.get('username') password = request.form.get('password') if username == 'wqbin' and password == '123456': return '登录成功' else: return self.__render(error='用户名或密码错误') app.add_url_rule('/login/',view_func=LoginView.as_view('login'))
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <form action="" method="post"> <table> <tbody> <tr> <td>用户名:</td> <td><input type="text" name="username"></td> </tr> <tr> <td>密码:</td> <td><input type="password" name="password"></td> </tr> <tr> <td></td> <td><input type="submit" value="立即登录"></td> </tr> </tbody> </table> {% if error %} <p style="color: red;">{{ error }}</p> {% endif %} </form> </body> </html>