04.视图进阶

add_url_rule和app.route原理剖析

add_url_rule

add_url_rule(rule,endpoint=None,view_func=None)

这个方法用来添加url与视图函数的映射,如果没有填写,那么默认会使用"view_func"的名字作为"endpoint",以后再使用"url_for"的时候,就要看在映射的时候有没有传递"endpoint"参数,如果传递了,那么就应该使用"endpoint"指定的字符串,如果没有传递,那么就应该使用"view_func"的名字。

  • view_func:视图函数

  • rule:路由

  • endpoint:可以配合url_for使用,并为url取一个名字

def test():
    return "这是一个测试,测试链接:{}".format(url_for("my_test"))

app.add_url_rule(rule="/test/",view_func=test,endpoint="my_test")
image-20200903035413849

app.route

这个装饰器,其实也是使用"add_url_rule"来实现url与视图函数映射的。

image-20200903035722213

注意使用了,endpoint之后,原本的url_for("index")就不能使用了,替换为了url_for("test")

标准类视图及其使用场景

类视图

之前的视图都是函数,所以一般简称为视图函数。其实视图也可以基于类来实现,类视图的好处是支持继承,但是类视图不能跟函数视图一样,写完类视图好需要通过 app.add_urlrule(urlrule,view_func)

以下将对两种类视图进行讲解:

标准视图

标准视图继承自flask.views.View,并且在子类中必须实现 dispatch_request方法,这个方法类似于视图函数,也要返回一个基于Response或者子类的对象,以下将用一个例子进行讲解:

app.py

# 通过add_url_rule添加类视图和url的映射,并且在as_view方法中指定该url的名称,方便url_for函数调用

test.html

image-20200903040927141

基于调度方法的视图:

Flask提供了另外一种类视图flask.views.MethodView,对每个HTTP方法执行不同的函数(映射到对应的方法的小写的同名方法上),还对 RESTful API尤其有用。

栗子:

image-20200903041401318

用类视图的一个缺陷就是比较难用装饰器来装饰,比如有时候需要做权限验证的时候,比如:

如果要在类视图上进行装饰,只能在as_view函数上进行装饰了,使用方式:

从flask 0.8 开始,还可以通过在类中添加decorators属性来实现对视图的装饰:

推荐使用这种方法

标准类视图

  1. 标准类视图,必须继承来自"flask.views.View"

  2. 必须实现"dipatch_request"方法,以后请求过来后,都会执行这个方法,这个方法的返回值就相当于是之前的函数视图一样,也必须返回"Response"或者子类的对象,或者是字符串,或者是元组

  3. 必须通过"app.add_url_rule(rule,endpoint,view_func)"来做url与视图的映射。'view_func'这个参数,需要使用类视图下的'as_view'类方法类转换:TestView.as_view('test')

  4. 如果指定了"endpoint",那么在使用"url_for"反转的时候就必须使用"endpoint"指定的那个值,如果没有指定"endpoint",那么就可以使用"as_view(视图名字)"中指定的视图名字作为反转

  5. 类视图有以下好处:可以继承,把一些共性的东西抽取出来放到父视图中,子视图直接拿来使用就可以了,但是也不是说所有的视图都要使用类视图,这个要根据情况而定

  • 字符串

image-20200903041815172
  • json格式

image-20200903042232514
  • 共同变量

image-20200903042744476
image-20200903042756942

类视图中使用装饰器

  • 如果使用的是函数视图,那么自己定义的装饰器必须放在"app.route"下面,否者装饰器失效。

image-20200903043128527
image-20200903043141698
  • 类视图的装饰器,需要重写类视图的一个类属性"decorators",这个类属性是一个列表或者元组都可以,里面装的是装饰器。

蓝图

作用

蓝图的作用就是让flask项目更加模块化,结构更加清晰,可以将相同模块的视图函数放在同一个蓝图下,同一个文件中,方便管理。

基本使用

  • 创建子目录

image-20200903043637958
  • 编写子目录文件的视图函数

如果想要某个蓝图下的所有url都有一个url前缀,那么可以定义蓝图的时候,指定url_prefix参数

demo1.py

demo2.py

  • 注册蓝图

app.py

  • 运行

image-20200903044242676
image-20200903044232266
image-20200903044220660

模版文件寻找规则

蓝图通过Blueprint函数中的template_folder参数进行修改模板文件的默认位置。

  • 如果项目中的templates文件夹中有相应的模板文件,就直接使用了

  • 如果项目中的templates文件夹中没有相应的模板文件,那么就到在定义蓝图的时候指定的路径中寻找,并且蓝图中的指定可以为相对路径,相对的是当前这个蓝图的文件所在的目录

如下:

image-20200903044648478

因为这个蓝图文件是在test3/demo1.py,那么就会到test3文件夹下的test_template文件夹中寻找模板文件

静态文件寻找规则

  • 在模板文件中,加载静态文件,如果使用url_for('static'),只会在app模板文件夹目录下查找静态文件

  • 如果在加载静态文件的时候,指定的蓝图的名字,比如:'demo1.static',那么就会到这个蓝图指定的static_forder下查找静态文件

如下:

目录

image-20200903045233825

demo1.py

test/test_template/demo1.html

test/test_static/demo1.css

url_for反转蓝图注意事项

  • 如果使用蓝图,那么以后想要反转蓝图中的视图函数为url,那么就应该在使用url_for的时候就指定这个蓝图。比如news.news_list,否则就找不到这个endpoint,在模板中的url_for同样也是满足这个条件,就是指定蓝图的名字。

  • 即使在同一蓝图中反转视图函数,也要指定蓝图的名字。

app.py

index.html

子域名实现详解

  • 使用蓝图技术

  • 在创建蓝图对象的时候,需要传递一个"subdomain"参数,来指定这个子域名的前缀,例如

image-20200903050329142
  • 需要在主app文件中,需要配置app.config的SERVER_NAME参数,例如:

  • 配置蓝图中子域名,通过subdomain参数实现

注意ip地址不能有子域名,localhost也不能有子域名

在"C:\Windows\System32\drivers\etc"文件中,打开hosts文件,然后添加域名与本机的映射,例如

image-20200903050627949
image-20200903050639412

最后更新于

这有帮助吗?