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 提供支持
在本页
  • URL与视图函数的映射
  • 传递参数
  • 设置参数
  • 接收用户传递的参数
  • url_for(反转url)
  • url_for的基本使用
  • 为什么需要用url_for
  • 自定义url转换器
  • 访问
  • 让其他电脑访问我的网站
  • 指定端口号
  • url唯一
  • 'GET'请求和'POST'请求
  • 重定向
  • 介绍
  • 函数
  • 视图函数Response
  • 关于响应
  • 自定义响应
  • Response函数
  • 自定义Reponse对象

这有帮助吗?

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

02.路由

URL与视图函数的映射

传递参数

传递参数的语法是:<参数名>,然后在视图函数中,也要定义同名的参数<type: variable>,其中type就是类型名称

  • string:默认的数据类型,接受没有任何斜杠"/"的文本

  • int:接受整型

  • float:接受浮点类型

  • path:和string的类似,但是接受斜杠

  • uuid:只接受uuid字符串(唯一,长度过长)。'uuid'只能接受符合'uuid'的字符擦混,'uuid'是一个唯一的字符串,用于做表的主键

  • any:可以指定多种路径。'any'数据类型可以在一个"url"中的指定多个路径。

import uuid
print(uuid.uuid4())

设置参数

  1. 固定参数

@app.route('/test')
def test():
    return "test"
  1. 不固定参数,定义参数类型

@app.route('/test2/<int:number>')
def test2(number):
    return "这是整数{}".format(number)

@app.route('/test3/<uuid:id>')
def test3(id):
	# uuid测试值:086bbc87-2bcd-4c81-b83d-9c58a306dc7c
    return "这是uuid{}".format(id)
  1. 多个路径

@app.route('/test4/<any(a,b):url_path>/<id>')
def test4(url_path,id):
    if url_path == 'a':
        return "这是{}:{}".format(url_path,id)
    else:
        return"这是{}:{}".format(url_path, id)

接收用户传递的参数

  • 第一种:使用path的形式(将参数嵌入到路径中),如上

  • 第二种:使用查询字符串的方式,传递的参数值形式类似url?key=value

from flask import Flask,request

app = Flask(__name__)

@app.route('/test5/')
def test5():
    wd = request.args.get("wd")
    return "查询字符串参数:%s" % wd

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

通过request.args.get()获取用户端传递过来的参数值。

  • 如果页面要做"SEO"优化,推荐使用"path"形式,如果不在搜索引擎优化,就是用查询字符串

url_for(反转url)

url_for的基本使用

url_for的第一个参数为视图函数的名字,后面的参数都是进行传递的参数,例如
url_for(view_name,arg1=1,arg2=2),view_name为视图函数名称,arg1与arg2则是进行传递的参数。

如下:

from flask import Flask,url_for

app = Flask(__name__)

@app.route('/<int:number>')
def index(number):
    return url_for("index",number=number)

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

为什么需要用url_for

  • 更方便地获取url地址,如果url函数名被改变了,也不会到处去修改

  • 'url_for'会自动的处理那些特殊的字符,不需要手动去处理

from flask import Flask,url_for

app = Flask(__name__)

@app.route('/')
def index():
    return url_for("index",number=1,next="/")

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

tips:在使用url的时候,不要直接url,应该配合url_for使用,来进行使用url地址

自定义url转换器

1.实现一个类,继承自"BaseConverter"

2.在自定义的类中,重写"regex",也就是这个变量的正则表达式

3.将自定义的类,映射到"app.url_map.Converters"

  • 判断path路径是否符合

from flask import Flask,url_for
from werkzeug.routing import BaseConverter

app = Flask(__name__)

# 创建一个关于满足于手机号码格式的类
class TelephoneConverter(BaseConverter):
    regex = r"1[34578]\d{9}"

app.url_map.converters["te1"] = TelephoneConverter

@app.route('/phone/<te1:phone>')
def my_phone(phone):
    return "我的手机号码:{}".format(phone)

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

其中TelephoneConverter为自定义类,app.url_map.converters["te1"] = TelephoneConverter,将自定义类的映射到app.url_map.converters中

  • 自定义参数

from flask import Flask
from werkzeug.routing import BaseConverter
app = Flask(__name__)

# 创建一个关于满足于手机号码格式的类
class TestConverter(BaseConverter):
    def to_python(self,value):
        return value

    def to_url(self,value):
        return value

app.url_map.converters["test1"] = TestConverter

@app.route('/test/<test1:value>')
def test(value):
    value = str(value).split('+')
    return "{}".format(value)

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

比如,想要将"a+b"字符串解析成“a”和“b”两个字符,通过split函数来进行解析,但是有很多函数都有这样的要求的话,每个函数都需要写上这一句,有点小麻烦,可以通过自定义类来机芯解决。

如下,看TestConvert类

from flask import Flask
from werkzeug.routing import BaseConverter
app = Flask(__name__)

# 创建一个关于满足于手机号码格式的类
class TestConverter(BaseConverter):
    def to_python(self,value):
        print(value)
        return str(value).split('+')

    def to_url(self,value):
        print(value)
        return '+'.join(value)

app.url_map.converters["test1"] = TestConverter

@app.route('/test/<test1:value>')
def test(value):
    return "{}".format(value)

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

其中

  • ’to_python‘的作用:这个方法的返回值,将会传递到view函数中作为参数

  • 'to_url'的作用:这个方法的返回值,将会在调用url_for函数的时候生成符合要求的URL形式

访问

让其他电脑访问我的网站

如果想在同一个局域网下的其他电脑访问自己的电脑上的Flask网站,那么可以设置'host="0.0.0.0"'才能访问得到

if __name__ == '__main__':
    app.run(host="0.0.0.0")

指定端口号

Flask项目,默认使用'5000'端口,如果想要更换端口,那么可以设置'port=9000'

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

url唯一

在定义url的时候,一定要记住在最后加一个斜杠

1.如果不加斜杠,那么在浏览器中访问这个url的时候,如果最后加了斜杠,那么就访问不到,这样用户体验不太好

2.搜索引擎会将不加斜杠的和加斜杠的示为两个不同的url,而其实加和不加斜杠的都是同一个url,那么就会搜索引擎造成一个误解,加了斜杠,就不会出现没有斜杠的情况

如下:

'GET'请求和'POST'请求

'GET'在网络请求中有许多请求方式,比如'GET','POST',DELETE,PUT请求等,那么最常用的就是'GET'和'POST'请求

  1. 'GET'请求:只会在服务上获取资源,不会更改服务器的状态。这种请求方式推荐使用'GET'请求

  2. 'POST'请求:会给服务器提交一些数据或者文件,一般POST请求是会对服务器的状态产生影响,那么这种请求推荐使用post请求

  3. 关于参数传递:

    • 'GET'请求把参数放到'url'中,通过'?xx=xxx'的形式传递,因为会把参数放到url中,这样会不很安全

    • 'POST'请求:把参数放到'Form Data'中,会把参数放到Form Data中,避免了被偷瞄的风险。可以通过抓包的形式进行分析,因为POST请求可以提交一些数据给服务器,比如可以发送文件,那么就增加了很大的风险。所以POST请求,对于那些有经验的黑客来讲,其实是更不安全

  4. 在'Flask'中,'route'方法,默认将只能使用'GET'的方式请求这个url,如果想要设置自己的请求方式,那么应该传递一个'POST'请求

@app.route('/login/',methods=["GET","POST"])
def login():
    if request.method == "GET":
        return render_template("login.html")
    else:
        # form = TestForm(request.form)
        # return "{}".format(form.data)
        return "post"

出现如下错误:

flask.debughelpers.FormDataRoutingRedirect: b'A request was sent to this URL...

由于flask对于url严格的要求 斜杠,因此检查路由是否正确。

或者

如下,添加strict_slashes=False,这样就不会严格检查。

@app.route('/login/',methods=["GET","POST"], strict_slashes=False)
def login():
    if request.method == "GET":
        return render_template("login.html")
    else:
        # form = TestForm(request.form)
        # return "{}".format(form.data)
        return "post"

重定向

介绍

重定向分为永久性重定向和暂时性重定向,在页面上体现的操作就是浏览器会重一个页面自动跳转另外一个页面,比如用户访问了一个需要权限的页面,但是该用户当前并没有登录,因此我们给它重定向到登录页面

永久性重定向:http的状态码是301,多用于旧网址就废弃了要转到一个新的网址确保用户的访问 www.jingdong.com ---> www.jd.com

暂时性重定向:http的状态码是302,表示页面的暂时性跳转。比如访问一个需要权限的网址,如果用户没有登录,应该重定向到登录页面,这种情况下,应该用暂时性重定向

在flask中,重定向是通过flask.redirect(location,code=302)这个函数来实现,location表示需要重定向的url,应该配合之前将的url_for()函数来使用,code表示采用那个重定向,默认是302即暂时性重定向,可以修改成301来实现永久性重定向

函数

redirect(location,code)
  • location:跳转的url地址

  • code:状态码

配合url_for使用

from flask import Flask, redirect, url_for

app = Flask(__name__)

@app.route('/test/')
def test():
    return "test"

@app.route('/test1/')
def test1():
    return redirect(url_for("test"))

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

视图函数Response

关于响应

视图函数的返回值会被自动转换一个响应对象,flask的转换逻辑如下:

  • 如果返回的是一个合法的响应对象,则直接返回

  • 如果返回的是一个字符串,那么flask会重新创建一个werkzeug.wrappers.Response对象,Response将给字符串作为主体,状态码是200,MIME类型为text/html,然后返回该Response对象

  • 如果返回的是一个元组,元组中的数据类型是(response.status.headers)。status值会覆盖默认的200状态码,headers可以是一个列表或者字典,作为额外的消息头

  • 如果以上条件都不满足,Flask会假设返回值是一个合法的wsgi应用程序

自定义响应

自定义响应必须满足三个条件:

  • 必须继承自Response类

  • 必须实现类方法force_type(cls,rv,environ=None)

  • 必须制定app.response_clss为自定义的Response

Response函数

返回类型

  • 可以返回字符串:返回的字符串其实是底层将这个字符串包装成了一个'Response'对象

  • 可以返回元组:元组的形式是(响应体,状态码,头部信息,返回的元组在底层包装成了一个'Response'对象)

  • 可以返回'Reponse'及其子类

例如:

# 返回内容
@app.route('/test/')
def test():
    return Response("123",status=200,mimetype='text/html')

# 设置cookie
@app.route('/list1/')
def list1():
    # return {"username":"miku"}
    # return ['a','b']
    # 以上数据不接回调,会报错
    resp = Response("list1")
    resp.set_cookie("miku","angle")
    return resp

# 添加响应头
@app.route('/list2/')
def list2():
    return "list2",200,{"X-NAME":"MIKU","Server":"windows2003"}

自定义Reponse对象

  1. 继承'Response'类

  2. 实现方法'force_type(cls,rv,envison=None)'

  3. 指定'app.response_class'为自定义的'Response'对象

  4. 如果视图返回的数据,不是字符串,也不是元组,也不是Response,那么就会将返回值传给'force_type',然后'force_type'的返回值返回给前端

from flask import Flask, Response, jsonify

app = Flask(__name__)

class JsonResponse(Response):

    @classmethod
    def force_type(cls,response,environ=None):
        """
        这个方法只有视图函数返回非字符,非元组,非Response对象才会调用
        :param response:
        :param environ:
        :return:
        response,视图函数的返回值
        """
        # 判断response是否为字典类型
        if isinstance(response,dict):
            response = jsonify(response)

        return super(JsonResponse,cls).force_type(response,environ)

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

# json,非字符,非元组
@app.route('/')
def test():
    return {"name":"bob"}

if __name__ == '__main__':
    app.run(debug=True)
上一页01.前言下一页03.模板

最后更新于3年前

这有帮助吗?

image-20200901190338488
image-20200901190426254
image-20200901190544823
image-20200901190620091
image-20200901190629108
image-20200901191642619
image-20200901191825839
image-20200901194831185
image-20200901195646079
image-20200901200023694
image-20200901200425913
image-20200901201029789
image-20200901201330630
image-20200901202141677