LogoLogo
  • README
  • 前端编程
    • 01 Node JS
    • 02-ES6详解
    • 03-NPM详解
    • 04-Babel详解
    • 05-前端模块化开发
    • 06-WebPack详解
    • 07-Vue详解
    • 08-Git详解
    • 09-微信小程序
  • 人工智能
    • 机器学习
      • 二次分配问题
      • 非负矩阵
      • 概率潜在语义分析
      • 概率图模型
      • 集成学习
      • 降维
      • 距离度量
      • 决策树
      • 逻辑回归
      • 马尔可夫决策过程
      • 马尔可夫链蒙特卡洛法
      • 朴素贝叶斯法
      • 谱聚类
      • 奇异值分解
      • 潜在狄利克雷分配
      • 潜在语义分析
      • 强化学习
      • 社区算法
      • 时间序列模型
      • 特征工程
      • 条件随机场
      • 图论基础
      • 线性分类
      • 线性回归
      • 信息论中的熵
      • 隐马尔科夫模型
      • 支持向量机
      • 主成分分析
      • EM算法
      • Hermite 矩阵的特征值不等式
      • k-means聚类
      • k近邻法
      • PageRank算法
    • 深度学习
      • Pytorch篇
        • 01-线性模型
        • 02-梯度下降法
        • 03-反向传播
        • 04-pytorch入门
        • 05-用pytorch实现线性回归
        • 06-logistic回归
        • 07-处理多维特征的输入
        • 08-加载数据集
        • 09-多分类问题
        • 10-卷积神经网络
        • 11-循环神经网络
    • 图神经网络
      • 图神经网络笔记01
        • 01-图(Graphs)的结构
        • 02-网络的性质和随机图模型
        • 03-网络工具
        • 04-网络中的主题和结构角色
        • 05-网络中的社区结构
      • 图神经网络笔记02
        • 01-深度学习引言
        • 02-神经网络基础
        • 03-卷积神经网络
        • 04-图信号处理与图卷积神经网络
        • 05-GNN的变体与框架-
        • [06-Google PPRGo 两分钟分类千万节点的最快GNN](人工智能/图神经网络/图神经网络笔记02/06-Google%20PPRGo 两分钟分类千万节点的最快GNN.md)
        • 07-序列模型
        • 08-变分自编码器
        • 09-对抗生成网络
  • 日常记录
    • 健身日记
    • 面经记录
    • 自动生成Summary文件
  • 实战项目
    • 谷粒商城
      • 00-项目概述
      • 01-分布式基础-全栈开发篇
      • 02-分布式高级-微服务架构篇
      • 03-高可用集群-架构师提升篇
  • 数据库
    • MySQL笔记
      • 01-MySQL基础篇
      • 02-MySQL架构篇
      • 03-MySQL索引及调优篇
      • 04-MySQL事务篇
      • 05-MySQL日志与备份篇
    • Redis笔记
      • 01-Redis基础篇
      • 02-Redis高级篇
    • 02-Redis篇
  • 算法笔记
    • 01-算法基础篇
    • 02-算法刷题篇
  • 职能扩展
    • 产品运营篇
  • Go编程
    • 01-Go基础
      • 01-Go基础篇
  • Java编程
    • 01-Java基础
      • 01-Java基础篇
      • 02-多线程篇
      • 03-注射与反解篇
      • 04-JUC并发编程篇
      • 05-JUC并发编程与源码分析
      • 06-JVM原理篇
      • 07-Netty原理篇
      • 08-设计模式篇
    • 02 Java Web
      • 01-Mybatis篇
      • 01-Mybatis篇(新版)
      • 02-Spring篇
      • 02-Spring篇(新版)
      • 03-SpringMVC篇
      • 04-MybatisPlus篇
    • 03-Java微服务
      • 01-SpringBoot篇
      • 01-SpringBoot篇(新版)
      • 02-SpringSecurity篇
      • 03-Shiro篇
      • 04-Swagger篇
      • 05-Zookeeper篇
      • 06-Dubbo篇
      • 07-SpringCloud篇
      • 08-SpringAlibaba篇
      • 09-SpringCloud篇(新版)
    • 04-Java中间件
      • 数据库篇
        • 01-分库分表概述
        • 02-MyCat篇
        • 03-MyCat2篇
        • 04-Sharding-jdbc篇
        • 05-ElasticSearch篇
      • 消息中间件篇
        • 01-MQ概述
        • 02-RabbitMQ篇
        • 03-Kafka篇
        • 04-RocketMQ篇
        • 05-Pulsar篇
    • 05-扩展篇
      • Dubbo篇
      • SpringBoot篇
      • SpringCloud篇
    • 06-第三方技术
      • 01-CDN技术篇
      • 02-POI技术篇
      • 03-第三方支付技术篇
      • 04-第三方登录技术篇
      • 05-第三方短信接入篇
      • 06-视频点播技术篇
      • 07-视频直播技术篇
    • 07-云原生
      • 01-Docker篇
      • 02-Kubernetes篇
      • 03-Kubesphere篇
  • Linux运维
    • 01-Linux篇
    • 02-Nginx篇
  • Python编程
    • 01-Python基础
      • 01.配置环境
      • 02.流程控制
      • 03.数值
      • 04.操作符
      • 05.列表
      • 06.元祖
      • 07.集合
      • 08.字典
      • 09.复制
      • 10.字符串
      • 11.函数
      • 12.常见内置函数
      • 13.变量
      • 14.异常和语法错误
      • 15.时间和日期
      • 16.正则表达式
    • 02 Python Web
      • flask篇
        • 01.前言
        • 02.路由
        • 03.模板
        • 04.视图进阶
        • 05.flask-sqlalchemy
        • 06.表单WTForms
        • 07.session与cookie
        • 08.上下文
        • 09.钩子函数
        • 10.flask 信号
        • 11.RESTFUL
        • 13.flask-mail
        • 14.flask+celery
        • 15.部署
        • 16.flask-login
        • 17.flask-cache
        • 18.flask-babel
        • 19.flask-dashed
        • 20.flask-pjax
        • 21.flask上传文件到第三方
        • 22.flask-restless
        • 23.flask-redis
        • 24.flask-flash
        • 25.消息通知
        • 26.分页
    • 03-Python数据分析
      • Matplotlib
      • Numpy
      • Pandas
      • Seaborn
    • 04-Python爬虫
      • 1.准备工作
      • 2.请求模块的使用
      • 3.解析模块的使用
      • 4.数据存储
      • 5.识别验证码
      • 6.爬取APP
      • 7.爬虫框架
      • 8.分布式爬虫
由 GitBook 提供支持
在本页
  • add_url_rule和app.route原理剖析
  • add_url_rule
  • app.route
  • 标准类视图及其使用场景
  • 类视图
  • 标准视图
  • 基于调度方法的视图:
  • 标准类视图
  • 类视图中使用装饰器
  • 蓝图
  • 作用
  • 基本使用
  • 模版文件寻找规则
  • 静态文件寻找规则
  • url_for反转蓝图注意事项
  • 子域名实现详解

这有帮助吗?

在GitHub上编辑
  1. Python编程
  2. 02 Python Web
  3. flask篇

04.视图进阶

上一页03.模板下一页05.flask-sqlalchemy

最后更新于3年前

这有帮助吗?

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

app.route(rule,**options)

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

@app.route('/',endpoint="test")
def index():
    return '测试链接1:{}'.format(url_for("test"))

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

标准类视图及其使用场景

类视图

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

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

标准视图

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

app.py

from flask import Flask, views, request, render_template

app = Flask(__name__)

class BaseView(views.View):

    # 自定义方法,用来获取模板路径
    def get_template_name(self):
        raise NotImplementedError()

    # 必须实现的方法,用来处理请求的
    def dispatch_request(self):
        if request.method != 'GET':
            return 'method error'

        # 这想从self.get_data()中获取函数,子类应该实现这个方法
        context = {'data': self.get_data()}
        return render_template(self.get_template_name(), **context)

class TestView(BaseView):
    # 实现从父类继承的获取模板路径的方法
    def get_template_name(self):
        return 'test.html'

    # 重写获取函数的方法
    def get_data(self):
        return [{
            'bool': 'false',
            'address': 'http://www.baidu.com',
        }]

# 通过add_url_rule添加类视图和url的映射,并且在as_view方法中指定该url的名称,方便url_for函数调用
app.add_url_rule('/test/',view_func=TestView.as_view('testview'))

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

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

test.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
这是一个测试html,测试值{data:{{ data }}}
</body>
</html>

基于调度方法的视图:

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

栗子:

class TestAPI(views.MethodView):
    #客户端通过get方法进行访问的时候执行的函数、
    def get(self):
        return jsonify({
        'name':'angle',
        'address':'http://www.baidu.com',
    })

    # 当客户端通过post方法进行访问的时候执行的函数
    def post(self):
        return 'post 请求'

app.add_url_rule(rule='/test/',view_func=TestAPI.as_view('test_api_view'))

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

def user_required(f)
    def decorator(*args,**kwargs):
        if not g.user:
            return 'auth failure'
        return f(*args,**kwargs)
    return decorator

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

view = user_required(UserAPI.as_view('users'))
app.add_url_rule('/users/',views_func=view)

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

class UserAPI(views.MethodView):
    decorator = [user_required]
    ....

推荐使用这种方法

标准类视图

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

  • 字符串

class TestView(views.View):

    def dispatch_request(self):
        return "test view"
# 类.as_view(name) 返回一个函数,name:视图名
# endpoint没指定,就是用view_func名
app.add_url_rule('/test/',endpoint='test',view_func=TestView.as_view('test'))
  • json格式

# 有几个url需要返回json数据,“需要用jsonify”
class JSONView(views.View):
    def get_data(self):
        raise NotImplementedError

    def dispatch_request(self):
        return jsonify(self.get_data())

# #将视图函数中返回的字典,转换成json对象,然后返回
# class JSONResponse(Response):
#
#     @classmethod
#     def force_type(cls, response, environ=None):
#         """
#         这个方法只有视图函数返回非字符,非元组,非Response对象才会调用
#         :param response:
#         :param environ:
#         :return:
#         response,视图函数的返回值
#         """
#         print(response)
#         if isinstance(response,dict):
#         # 转换
#             response = jsonify(response)
#         return super(JSONResponse,cls).force_type(response,environ)

# 添加response类,一定要添加
# app.response_class = JSONResponse

# 必须继承自views
class TestView(JSONView):
    def get_data(self):
        return {"username":"angle"}

app.add_url_rule('/test/',endpoint='test',view_func=TestView.as_view('test'))
  • 共同变量

from flask import Flask, views, request, render_template, jsonify
import json

app = Flask(__name__)

class ADSView(views.View):
    def __init__(self):
        super(ADSView,self).__init__()
        self.context = {
            'ads':"广告"
        }

# 提取共同变量
class LoginView(ADSView):
    def dispatch_request(self):
        self.context.update({
            'username':'login',
        })
        # 保证字典的中文不乱码
        return json.dumps(self.context, ensure_ascii=False)

class RegistView(ADSView):
    def dispatch_request(self):
        return json.dumps(self.context, ensure_ascii=False)

app.add_url_rule('/ads/',endpoint='ads',view_func=ADSView.as_view('ads'))
app.add_url_rule('/login/',endpoint='login',view_func=LoginView.as_view('login'))
app.add_url_rule('/regist/',endpoint='regist',view_func=RegistView.as_view('regist'))

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

类视图中使用装饰器

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

from flask import Flask,request
from functools import wraps

app = Flask(__name__)

def login_required(func):
    @wraps(func)
    def wrapper(*args,**kwargs):
        username  = request.args.get("username")
        if username and username == "code":
            return func(*args,**kwargs)
        else:
            return "请先登录"
    return wrapper

@app.route('/settings/')
@login_required
def settings():
    return "这是设置界面"

if __name__ == '__main__':
    app.run(debug=True)
  • 类视图的装饰器,需要重写类视图的一个类属性"decorators",这个类属性是一个列表或者元组都可以,里面装的是装饰器。

class ProfileView(views.View):

    # 装饰器
    decorators = [login_required]

    def dispatch_request(self):
        return "个人中心界面"

app.add_url_rule(rule='/profile/',view_func=ProfileView.as_view('profile'))

蓝图

作用

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

基本使用

  • 创建子目录

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

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

demo1.py

from flask import Blueprint

demo1_bp = Blueprint("demo1",__name__,url_prefix='/demo1')

@demo1_bp.route('/')
def index():
    return "this is demo1"

demo2.py

from flask import Blueprint

demo2_bp = Blueprint("/demo2",__name__,url_prefix='/demo2')

@demo2_bp.route('/')
def index():
    return "this is demo2"
  • 注册蓝图

app.py

from flask import Flask,Blueprint
from test3.demo1 import demo1_bp
from test3.demo2 import demo2_bp

app = Flask(__name__)

app.register_blueprint(demo1_bp)
app.register_blueprint(demo2_bp)

@app.route('/')
def index():
    return "this is app"

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

模版文件寻找规则

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

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

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

如下:

demo1_bp = Blueprint("demo1",__name__,url_prefix='/demo1',template_folder="test_template")

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

静态文件寻找规则

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

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

如下:

目录

demo1.py

from flask import Blueprint, render_template

demo1_bp = Blueprint("demo1",__name__,url_prefix='/demo1',template_folder="test_template",static_folder="test_static")

@demo1_bp.route('/')
def index():
    return render_template('demo1.html')

test/test_template/demo1.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <link rel="stylesheet" href="{{ url_for('demo1.static',filename='demo1.css')}}">
</head>
<body>
<button>点击</button>
</body>
</html>

test/test_static/demo1.css

button{
    background-color: yellow;
}

url_for反转蓝图注意事项

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

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

app.py

@app.route('/')
def index():
    return render_template("index.html")

index.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<a href="{{ url_for('demo1.index') }}">点击</a>
</body>
</html>

子域名实现详解

  • 使用蓝图技术

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

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

app.config['SERVER_NAME'] = 'test.com:5000'
  • 配置蓝图中子域名,通过subdomain参数实现

demo1_bp = Blueprint("demo1",__name__,
                     subdomain='demo1',
                     url_prefix='/demo1',
                     template_folder="test_template",
                     static_folder="test_static")

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

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

127.0.0.1 test.com
127.0.0.1 demo1.test.com
image-20200903035722213
image-20200903040927141
image-20200903041401318
image-20200903041815172
image-20200903042232514
image-20200903042744476
image-20200903042756942
image-20200903043128527
image-20200903043141698
image-20200903043637958
image-20200903044242676
image-20200903044232266
image-20200903044220660
image-20200903044648478
image-20200903045233825

image-20200903050329142
image-20200903050627949
image-20200903050639412