Flask笔记(一)
first_flask_project.py
# 从flask这个包中导入Flask这个类
# Flask这个类是项目的核心,以后很多操作都是基于这个类的对象
# 注册url、注册蓝图等都是基于这个类的对象
from flask import Flask
# 创建一个Flask对象,传递__name__参数进去
# __name__参数的作用:
# 1. 可以规定模版和静态文件的查找路径
# 2. 以后一些Flask插件,比如Flask-migrate、Flask-SQLAlchemy如果报错了,那么Flask可以通过这个参数找到具体的错误位置
app = Flask(__name__)
# @app.route:是一个装饰器
# @app.route('/')就是将url中的 / 映射到hello_world这个视图函数上面
# 以后你访问我这个网站的/目录的时候,会执行hello_world这个函数,然后将这个函数的返回值返回给浏览器。
# wwww.baidu.com/ -> hello_world函数
@app.route('/')
def hello_world():
return '知了课堂'
# www.baidu.com/list/
@app.route('/list/')
def my_list():
return 'my list'
# 如果这个文件是作为一个主文件运行,那么就执行app.run()方法
# 也就是启动这个网站
if __name__ == '__main__':
# app.run():Flask中的一个测试应用服务器
# while True:
# listen()
# input()
app.run(port=8000)
DEBUG模式:
为什么需要开启DEBUG模式:
- 如果开启了DEBUG模式,那么在代码中如果抛出了异常,在浏览器的页面中可以看到具体的错误信息,以及具体的错误代码位置。方便开发者调试。
- 如果开启了DEBUG模式,那么以后在
Python
代码中修改了任何代码,只要按ctrl+s
,flask
就会自动的重新记载整个网站。不需要手动点击重新运行。
配置DEBUG模式的四种方式:
- 在
app.run()
中传递一个参数debug=True
就可以开启DEBUG
模式。 - 给
app.deubg=True
也可以开启debug
模式。 - 通过配置参数的形式设置DEBUG模式:
app.config.update(DEBUG=True)
。 - 通过配置文件的形式设置DEBUG模式:
app.config.from_object(config)
。
PIN码:
如果想要在网页上调试代码,那么应该输入PIN
码。
debug_demo.py
from flask import Flask
import config
app = Flask(__name__)
# app.debug = True # 第二种方式
# app.config.update(DEBUG=True) # 第三种方式
# dict
# a = {'a':1}
# b = {'b':2}
# a.update(b=2)
# print(a)
# 设置配置参数的形式
# DEBUG必须要大写,不能小写
app.config.from_object(config) # 第四种方式
@app.route('/')
def hello_world():
a = 1
b = 0
print('hello zhiliao')
c = a/b
return 'Hello World!'
if __name__ == '__main__':
app.run()
# app.run(debug=True) # 第一种方式
config.py
#encoding: utf-8
DEBUG=True
config笔记:
使用app.config.from_object
的方式加载配置文件:
- 导入
import config
。 - 使用
app.config.from_object(config)
。
使用app.config.from_pyfile
的方式加载配置文件:
这种方式不需要import
,直接使用app.config.from_pyfile('config.py')
就可以了。
注意这个地方,必须要写文件的全名,后缀名不能少。
- 这种方式,加载配置文件,不局限于只能使用
py
文件,普通的txt
文件同样也适合。 - 这种方式,可以传递
silent=True
,那么这个静态文件没有找到的时候,不会抛出异常。
config_demo.py
from flask import Flask
app = Flask(__name__)
# app.config.from_object*(config) # 方式一(此方式需要导入 import config)
# app.config.from_pyfile('config.py') # 方式二(此方式不需要导入)
app.config.from_pyfile('config.tx',silent=True) # 方式二(silent=True,这个静态文件没有找到的时候,不会抛出异常。若为Flase,则会抛出异常。)
@app.route('/')
def hello_world():
return 'Hello World!'
if __name__ == '__main__':
app.run()
config.txt
DEBUG=True
config.py
#encoding: utf-8
DEBUG = True
URL与视图函数的映射:
传递参数:
传递参数的语法是:/<参数名>/
。然后在视图函数中,也要定义同名的参数。
参数的数据类型:
- 如果没有指定具体的数据类型,那么默认就是使用
string
数据类型。 int
数据类型只能传递int
类型。float
数据类型只能传递float
类型。path
数据类型和string
有点类似,都是可以接收任意的字符串,但是path
可以接收路径,也就是说可以包含斜杠。uuid
数据类型只能接收符合uuid
的字符串。uuid
是一个全宇宙都唯一的字符串,一般可以用来作为表的主键。any
数据类型可以在一个url
中指定多个路径。例如:@app.route('/<any(blog,article):url_path>/<id>/')
def detail(url_path,id):
if url_path == 'blog':
return '博客详情:%s' % id
else:
return '博客详情:%s' % id
接收用户传递的参数:
- 第一种:使用path的形式(将参数嵌入到路径中),就是上面讲的。
- 第二种:使用查询字符串的方式,就是通过
?key=value
的形式传递的。@app.route('/d/')
def d():
wd = request.args.get('wd')
return '您通过查询字符串的方式传递的参数是:%s' % wd
- 如果你的这个页面的想要做
SEO
优化,就是被搜索引擎搜索到,那么推荐使用第一种形式(path的形式)。如果不在乎搜索引擎优化,那么就可以使用第二种(查询字符串的形式)。
url_view_demo.py
from flask import Flask,request
app = Flask(__name__)
@app.route('/')
def hello_world():
return 'Hello World!'
@app.route('/list/')
def article_list():
return 'article list'
@app.route('/p/<float:article_id>')
def article_detail(article_id):
return '您请求的文章是:%s' % article_id
# @app.route('/article/<path:test>/')
# def test_article(test):
# return 'test article:%s' % test
@app.route('/u/<uuid:user_id>/')
def user_detail(user_id):
return '用户个人中心页面:%s' % user_id
# /blog/<id>/
# /user/<id>/
@app.route('/<any(blog,article):url_path>/<id>/')
def detail(url_path,id):
if url_path == 'blog':
return '博客详情:%s' % id
else:
return '博客详情:%s' % id
# 通过问号的形式传递参数
@app.route('/d/')
def d():
wd = request.args.get('wd')
ie = request.args.get('ie')
print('ie:',ie)
return '您通过查询字符串的方式传递的参数是:%s' % wd
# import uuid
# print(uuid.uuid4())
if __name__ == '__main__':
app.run(debug=True)
url_for笔记:
url_for
的基本使用:
url_for
第一个参数,应该是视图函数的名字的字符串。后面的参数就是传递给url
。
如果传递的参数之前在url
中已经定义了,那么这个参数就会被当成path
的形式给
url
。如果这个参数之前没有在url
中定义,那么将变成查询字符串的形式放到url
中。
@app.route('/post/list/<page>/')
def my_list(page):
return 'my list'
print(url_for('my_list',page=1,count=2))
# 构建出来的url:/my_list/1/?count=2
为什么需要url_for
:
- 将来如果修改了
URL
,但没有修改该URL对应的函数名,就不用到处去替换URL了。 url_for
会自动的处理那些特殊的字符,不需要手动去处理。url = url_for('login',next='/')
# 会自动的将/编码,不需要手动去处理。
# url=/login/?next=%2F
强烈建议以后在使用url的时候,使用url_for
来反转url。
url_for_demo.py
from flask import Flask,url_for,request
app = Flask(__name__)
# / => hello_world
# hello_world => /
@app.route('/')
def hello_world():
print(url_for('my list',page=1)) # 类似这样
# /list/
# /list/1/
# return 'Hello World'
return url_for('login',next='/')
# /login/?next=/
# return '/post/list/1/'
@app.route('/login/')
def login():
return 'login'
@app.route('/post/list/<page>/')
def my_list(page):
return 'my list'
@app.route('/post/detail/<id>/')
def detail():
return 'detail'
if __name__ == '__main__':
app.run(debug=True)
自定义URL转换器:
自定义URL转换器的方式:
- 实现一个类,继承自
BaseConverter
。 - 在自定义的类中,重写
regex
,也就是这个变量的正则表达式。 - 将自定义的类,映射到
app.url_map.converters
上。比如:app.url_map.converters['tel'] = TelephoneConverter
to_python
的作用:
这个方法的返回值,将会传递到view函数中作为参数。
to_url
的作用:
这个方法的返回值,将会在调用url_for函数的时候生成符合要求的URL形式。
custom_url_converter.py
from flask import Flask,url_for
from werkzeug.routing import BaseConverter
app = Flask(__name__)
# 一个url中,含有手机号码的变量,必须限定这个变量的字符串格式满足手机号码的格式
class TelephoneConveter(BaseConverter):
regex = r'1[85734]\d{9}'
# 用户在访问/posts/a+b/
class ListConverter(BaseConverter):
def to_python(self, value):
return value.split('+')
def to_url(self, value):
return "+".join(value)
# return "hello"
app.url_map.converters['tel'] = TelephoneConveter
app.url_map.converters['list'] = ListConverter
@app.route('/')
def hello_world():
print('='*30)
print(url_for('posts',boards=['a','b']))
print('='*30)
return 'Hello World!'
@app.route('/user/<string:user_id>/')
def user_profile(user_id):
return '您输入的user_id为:%s' % user_id
@app.route('/telephone/<tel:my_tel>/')
def my_tel(my_tel):
return '您的手机号码是:%s' % my_tel
@app.route('/posts/<list:boards>/')
def posts(boards):
print(boards)
return "您提交的板块是:%s" % boards
if __name__ == '__main__':
app.run(debug=True)
必会的小细节知识点:
在局域网中让其他电脑访问我的网站:
如果想在同一个局域网下的其他电脑访问自己电脑上的Flask网站,
那么可以设置host='0.0.0.0'
才能访问得到。
指定端口号:
Flask项目,默认使用5000
端口。如果想更换端口,那么可以设置port=9000
。
url唯一:
在定义url的时候,一定要记得在最后加一个斜杠。
- 如果不加斜杠,那么在浏览器中访问这个url的时候,如果最后加了斜杠,那么就访问不到。这样用户体验不太好。
- 搜索引擎会将不加斜杠的和加斜杠的视为两个不同的url。而其实加和不加斜杠的都是同一个url,那么就会给搜索引擎造成一个误解。加了斜杠,就不会出现没有斜杠的情况。
GET
请求和POST
请求:
在网络请求中有许多请求方式,比如:GET、POST、DELETE、PUT请求等。那么最常用的就是GET
和POST
请求了。
GET
请求:只会在服务器上获取资源,不会更改服务器的状态。这种请求方式推荐使用GET
请求。POST
请求:会给服务器提交一些数据或者文件。一般POST请求是会对服务器的状态产生影响,那么这种请求推荐使用POST请求。关于参数传递:
GET
请求:把参数放到url
中,通过?xx=xxx
的形式传递的。因为会把参数放到url中,所以如果视力好,一眼就能看到你传递给服务器的参数。这样不太安全。POST
请求:把参数放到Form Data
中。会把参数放到Form Data
中,避免了被偷瞄的风险,但是如果别人想要偷看你的密码,那么其实可以通过抓包的形式。因为POST请求可以提交一些数据给服务器,比如可以发送文件,那么这就增加了很大的风险。所以POST请求,对于那些有经验的黑客来讲,其实是更不安全的。
在
Flask
中,route
方法,默认将只能使用GET
的方式请求这个url,如果想要设置自己的请求方式,那么应该传递一个methods
参数。
url_detail.py
from flask import Flask,request,render_template
app = Flask(__name__)
@app.route('/')
def hello_world():
return 'Hello World!'
@app.route('/list/',methods=['POST'])
def my_list():
return 'list'
@app.route('/login/',methods=['GET','POST'])
def login():
if request.method == 'GET':
return render_template('login.html')
else:
return 'success'
if __name__ == '__main__':
# host
app.run(debug=True,host='0.0.0.0')
重定向笔记:
重定向分为永久性重定向和暂时性重定向,在页面上体现的操作就是浏览器会从一个页面自动跳转到另外一个页面。比如用户访问了一个需要权限的页面,但是该用户当前并没有登录,因此我们应该给他重定向到登录页面。
永久性重定向:
http
的状态码是301
,多用于旧网址被废弃了要转到一个新的网址确保用户的访问,最经典的就是京东网站,你输入www.jingdong.com
的时候,会被重定向到www.jd.com
,因为jingdong.com
这个网址已经被废弃了,被改成jd.com
,所以这种情况下应该用永久重定向。暂时性重定向:
http
的状态码是302
,表示页面的暂时性跳转。比如访问一个需要权限的网址,如果当前用户没有登录,应该重定向到登录页面,这种情况下,应该用暂时性重定向。
flask中重定向:
flask
中有一个函数叫做redirect
,可以重定向到指定的页面。示例代码如下:
from flask import Flask,request,redirect,url_for
app = Flask(__name__)
@app.route('/login/')
def login():
return '这是登录页面'
@app.route('/profile/')
def profile():
if request.args.get('name'):
return '个人中心页面'
else:
# redirect 重定向
return redirect(url_for('login'))
redirect_demo.py
from flask import Flask,request,redirect,url_for
app = Flask(__name__)
@app.route('/')
def hello_world():
return 'Hello World!'
@app.route('/login/')
def login():
return '这是登录页面'
@app.route('/profile/')
def profile():
if request.args.get('name'):
return '个人中心页面'
else:
return redirect(url_for('login'))
if __name__ == '__main__':
app.run()
response笔记:
视图函数中可以返回哪些值:
- 可以返回字符串:返回的字符串其实底层将这个字符串包装成了一个
Response
对象。 - 可以返回元组:元组的形式是(响应体,状态码,头部信息),也不一定三个都要写,写两个也是可以的。返回的元组,其实在底层也是包装成了一个
Response
对象。 - 可以返回
Response
及其子类。
实现一个自定义的Response
对象:
- 继承自
Response
类。 - 实现方法
force_type(cls,rv,environ=None)
。 - 指定
app.response_class
为你自定义的Response
对象。 - 如果视图函数返回的数据,不是字符串,也不是元组,也不是Response对象,那么就会将返回值传给
force_type
,然后再将force_type
的返回值返回给前端。
response_demo.py
from flask import Flask,Response,jsonify,render_template
# flask = werkzeug+sqlalchemy+jinja2
import json
app = Flask(__name__)
# 将视图函数中返回的字典,转换成json对象,然后返回
# restful-api
class JSONResponse(Response):
@classmethod
def force_type(cls, response, environ=None):
"""
这个方法只有视图函数返回非字符、非元组、非Response对象
才会调用
response:视图函数的返回值
"""
if isinstance(response,dict):
# jsonify除了将字典转换成json对象,还将改对象包装成了一个Response对象
response = jsonify(response)
return super(JSONResponse, cls).force_type(response,environ)
app.response_class = JSONResponse
@app.route('/')
def hello_world():
# Response('Hello World!',status=200,mimetype='text/html')
# return 'Hello World!'
return render_template('index.html')
@app.route('/list1/')
def list1():
resp = Response('list1')
resp.set_cookie('country','china')
return resp
@app.route('/list2/')
def list2():
return 'list2',200,{'X-NAME':'zhiliao'}
@app.route('/list3/')
def list3():
return {'username':'zhiliao','age':18}
if __name__ == '__main__':
app.run(debug=True,port=5000)
templates/index.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
hello world
</body>
</html>
Flask笔记(一)的更多相关文章
- flask笔记(三)Flask 添加登陆验证装饰器报错,及解析
Flask 添加登陆验证装饰器报错,及解析 写这个之前,是想到一个需求,这个是关于之前写Flask笔记(二)中的一个知识点,路由相关 需求为 : 有一些页面必须是登陆之后才能访问的,比如Shoppin ...
- Flask笔记1
Flask笔记 首先明确一下,要运行一个动态网页,我们需要 一个 Web 服务器来监听并响应请求,如果请求的是静态文件它就直接将其返回,如果是动态 url 它就将请求转交给 Web 应用. 一个 We ...
- Flask笔记:RESTful
RESTful是用于前台和后端进行通信的一种规范或者说一种风格,采用的是HTTP和HTTPS协议,数据传输的格式使用的都是JSON,而不是XML.通常,RESTful的URL中只有名词,没有动词,而且 ...
- flask笔记3-模板
flask框架使用jinja2模板引擎.简单的说,模板就是一个纯html文件中夹杂着占位符,在渲染模板时用真实变量值替换占位符,就形成了最终的前台页面. 1.模板存放位置: 默认情况下,flask在根 ...
- flask笔记2-程序的基本结构
第一个flask web程序 1.初始化(所有flask程序都必须创建一个程序实例,程序实例是Flask类的对象): from flask import Flask app = Flask(__nam ...
- flask笔记1-安装
1.创建应用的根目录: 2.在根目录下创建独立的虚拟python运行环境,创建完成后当前目录会有一个文件夹,即该独立环境(使用--no-site-pachages参数将不会复制任何系统python环境 ...
- flask笔记---url、变量规则
1.路由: route() 装饰器用于把一个函数绑定到一个 URL,可以动态变化 URL 的某些部分,还可以为一个函数指定多个规则,从而方便用户访问与记忆. 例子: @app.route('/') # ...
- flask笔记二
web表单 web表单是浏览者和网之间的一个互动平台,完成浏览器和服务器之间的数据交互. 1.用Flask-WTF来处理表单 (1)在根目录下编辑扩展配置--config.py CSRF_ENABLE ...
- flask笔记一
最近学习flask,由于web开发方面接触的并不是很多,所以看官方文档有点焦头烂额,好多的概念不理解. <Flask web 开发>比较基础,先用这本书做个入门. 1.Flask实例化对象 ...
随机推荐
- 【prometheus 抓取源】
配置prometheus从prometheus爬取数据 prometheus提供了下面这个端口来让其他prometheus来抓取(scrape)自己的时序数据: http://prometheus_i ...
- elasticsearch in语句和not in语句
sql语句示例: select * from table where t_id in (1,2,3,4) php代码示例: $search_query = [ "bool" =&g ...
- DevOps - DevOps精要 - 歧途
前言 如果在实施DevOps的过程中,周围没有一个人支持你,也没有得到领导和团队成员的理解: 如果在采用DevOps的工具和方法之后,难以获得明显的效率提升,甚至得到了不少的消极反馈: 那就需要反省一 ...
- [计算机视觉][ARM-Linux开发]OpenCV 3.1下载 ippicv_linux_20151201失败
安装OpenCV 3.1的过程中要下载ippicv_linux_20151201,由于网络的原因,这个文件经常会下载失败. 解决的办法是手动下载: 先下载 OpenCV 3.1 Download MD ...
- Meerkat软件
一.准备工作 meerkat 0.189版本和以前的版本相比,支持bwa mem 输出的bam文件,还支持全外显子数据count SV. meerkat原理 1.1 需要准备的软件 unix/Linu ...
- jupyter notebook选择conda环境
需要安装: conda install ipykernel 使用:首先激活对应的conda环境 source activate 环境名称 将环境写入notebook的kernel中 python -m ...
- Hystrix【入门】
公共依赖配置: <parent> <groupId>org.springframework.boot</groupId> <artifactId>spr ...
- Python 容器用法整理
本文整理几种基本容器:列表.元组.字典和集合的用法和collections中几种已经预先实现的容器数据结构:namedtuple(),双向链表deque,ChainMap,Counter,Ordere ...
- FZU2018级算法第五次作业 missile(排序+枚举)
在解题报告之前,首先对同一次作业中另外一题(求逆序对)某人在未经冰少允许情况下,擅自登录冰少账号,原模原样剽窃冰少代码,并且最终还被评为优秀作业的行为表示严正抗议! 题目大意: 二维平面上给出 n 个 ...
- WUSTOJ 1311: 开心的金明(Java)动态规划-01背包
题目链接: