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实例化对象 ...
随机推荐
- 关于Js异常
一.Javascript的异常处理机制 当javascript代码中出现错误的时候,js引擎就会根据js的调用栈逐级寻找对应的catch,如果没有找到相应的catch handler或catch ha ...
- SpringBoot学习笔记:读取配置文件
SpringBoot学习笔记:读取配置文件 配置文件 在以往的项目中,我们主要通过XML文件进行框架配置,业务的相关配置会放在属性文件中,然后通过一个属性读取的工具类来读取配置信息.在SpringBo ...
- UN Report: Last 10 Years Likely the Hottest Decade on Record——VOA慢速英语
听力地址:UN Report: Last 10 Years Likely the Hottest Decade on Record 中英对照:联合国报告称过去十年可能是有记录以来最热的十年 Words ...
- PCA分析的疑问
R 与python scikit-learn PCA的主成分结果有部分是反的 通过R和python分别计算出来的PCA的结果存在某些主成分的结果是相反的,这些结果是没有问题的,只是表示这个分量被反转了 ...
- LeetCode 198. 打家劫舍(House Robber) 5
198. 打家劫舍 198. House Robber 题目描述 你是一个专业的小偷,计划偷窃沿街的房屋.每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两 ...
- LeetCode 783. 二叉搜索树结点最小距离(Minimum Distance Between BST Nodes)
783. 二叉搜索树结点最小距离 LeetCode783. Minimum Distance Between BST Nodes 题目描述 给定一个二叉搜索树的根结点 root, 返回树中任意两节点的 ...
- 用 Python 把微信小程序的二维码转化成图片
官方文档 import cString import requests from tornado.web import authenticated, RequestHandler URL = 'htt ...
- mysql 添加大量测试数据
mysql 添加大量测试数据 场景 针对于大量测试数据插入,检测sql执行速度 第一步:建表 // 测试表 CREATE TABLE user ( id int(11) NOT NULL AUTO_I ...
- java并发知识点
前言 先列出java并发涉及的知识点,后面再慢慢补. java并发 1 常见概念 进程/线程 进程:程序执行的实体,操作系统资源调度资源分配的基本单元 线程:程序执行的最小单元,拥有独立的堆栈和局部变 ...
- SPA项目首页导航+左侧菜单
Mock.js是个啥 前后端分离之后,前端迫切需要一种机制,不再需要依赖后端接口开发,而今天的主角mockjs就可以做到这一点 Mock.js是一个模拟数据的生成器,用来帮助前端调试开发.进行前后端的 ...