前言
Python Web框架里比较有名当属Django,Django功能全面,它提供一站式解决方案,集成了MVT(Model-View-Template)和ORM,以及后台管理。但是缺点也很明显,它偏重。就像是一个装潢好的房子,它提供好了你要用的东西,直接拿来用就可以。
Flask相对于Django而言是轻量级的Web框架。和Django不同,Flask轻巧、简洁,通过定制第三方扩展来实现具体功能。
可定制性,通过扩展增加其功能,这是Flask最重要的特点。Flask的两个主要核心应用是Werkzeug和模板引擎Jinja。
WSGI
Web Server Gateway Interface(Web服务器网关接口,WSGI)已被用作Python Web应用程序开发的标准。 WSGI是Web服务器和Web应用程序之间通用接口的规范。
Werkzeug
它是一个WSGI工具包,它实现了请求,响应对象和实用函数。这使得能够在其上构建web框架。 Flask框架使用Werkzeug作为其基础之一。
jinja2
jinja2是Python的一个流行的模板引擎。Web模板系统将模板与特定数据源组合以呈现动态网页。
Flask通常被称为微框架。 它旨在保持应用程序的核心简单且可扩展。Flask没有用于数据库处理的内置抽象层,也没有形成验证支持。相反,Flask支持扩展以向应用程序添加此类功能。一些受欢迎的Flask扩展将在本教程后续章节进行讨论。
Flask工作流程图
Flask扩展包:
- Flask-SQLalchemy:操作数据库
- Flask-migrate:管理迁移数据库
- Flask-Mail:邮件
- Flask-WTF:表单
- Flask-script:插入脚本
- Flask-Login:认证用户状态
- Flask-RESTful:开发REST API的工具
- Flask-Bootstrap:集成前端Twitter Bootstrap框架
- Flask-Moment:本地化日期和时间
新建flask项目:
建成如下:
运行如下:
必须在项目中导入Flask模块。 Flask类的一个对象是我们的WSGI应用程序。
Flask构造函数使用当前模块(__name __)的名称作为参数。
以下是实例化一个Flask对象可填的参数,及其默认值:
- template_folder:模板所在文件夹的名字, 默认就是templates
- root_path:可以不用填,会自动找到,当前执行文件,所在目录地址在return render_template时会将上面两个进行拼接,找到对应的模板地址
- static_folder:静态文件所在文件的名字,默认是static,可以不用填
- static_url_path:静态文件的地址前缀,写成什么,访问静态文件时,就要在前面加上这个
instance_path和instance_relative_config是配合来用的、这两个参数是用来找配置文件的,当用app.config.from_pyfile('settings.py')这种方式导入配置文件的时候会用到
- instance_relative_config:默认为False,当设置为True时,from_pyfile会从instance_path指定的地址下查找文件。
- instsnce_path:指定from_pyfile查询文件的路径,不设置时,默认寻找和app.run()的执行文件同级目录下的instance文件夹,如果配置了instance_path(注意需要是绝对路径),就会从指定的地址下里面的文件
Flask类的route()函数是一个装饰器,它告诉应用程序哪个URL应该调用相关的函数。
app.route(rule, options)
- rule 参数表示与该函数的URL绑定。
- options 是要转发给基础Rule对象的参数列表。
在上面的示例中,'/ ' URL与hello_world()函数绑定。因此,当在浏览器中打开web服务器的主页时,将呈现该函数的输出。
application对象的add_url_rule()函数也可用于将URL与函数绑定,如下:
def hello_world():
return ‘hello world’
app.add_url_rule(‘/’, ‘hello’,hello_world)
最后,Flask类的run()方法在本地开发服务器上运行应用程序。
app.run(host, port, debug, options)
- host:监听的主机名
- post:监听的主机端口,默认5000
- debug:调试模式,默认false,设置为frue则显示调试信息
- options:要转发到底层的werkzeng服务器
给路由传参示例:
from flask importFlask app = Flask(__name__) @app.route('/id/<int:idn>')def hello_world(idn): return 'HelloWorld! %d' %idn if __name__ == '__main__': app.run()
路由传递的参数默认当做string处理,这里指定int,还支持float,path,尖括号中冒号后面的内容是动态的。
效果如下:
返回状态码示例:
abort函数:
如果在视图函数执行过程中,出现了异常错误,我们可以使用abort函数立即终止视图函数的执行。通过abort函数,可以向前端返回一个http标准中存在的错误状态码,表示出现的错误信息。
使用abort抛出一个http标准中不存在的自定义的状态码,没有实际意义。如果abort函数被触发,其后面的语句将不会执行。其类似于python中raise。
from flask importFlask,abort app = Flask(__name__) @app.route('/id/<int:idn>')def hello_world(idn): abort(403) return 'HelloWorld! %d' %idn, 888 if __name__ == '__main__': app.run()
在Flask中通过装饰器来实现捕获异常,errorhandler()接收的参数为异常状态码。视图函数的参数,返回的是错误信息。
重定向redirect示例:
from flask importFlask,redirect app = Flask(__name__) @app.route('/')def hello_world(): return redirect('https://www.baidu.com') if __name__ == '__main__': app.run()
效果如下:
这里再介绍一下url_for(),url_for()函数对于动态构建特定函数的URL非常有用。该函数接受函数的名称作为第一个参数,以及一个或多个关键字参数,每个参数对应于URL的变量部分。
如下:
@app.route('/user/<name>')def hello_user(name): if name=='admin': return redirect(url_for('hello_admin')) else: return redirect(url_for('hello_guest',guest =name))
HTTP方法:
Http协议是万维网中数据通信的基础。在该协议中定义了从指定URL检索数据的不同方法。
方法介绍:
- GET:以未加密的形式将数据发送到服务器获取响应内容
- POST: 用于将HTML表单数据上传到服务器
- PUT:用上传的内容替换目标资源的数据
- DELETe:删除由URL给出的目标资源的数据
- HEAD: 和GET方法相同,但没响应体
默认情况下,Flask路由响应GET请求。但是,可以通过为route()装饰器提供方法参数来更改此首选项。
为了演示在URL路由中使用POST方法,首先让我们创建一个HTML表单,并使用POST方法将表单数据发送到URL。
login.html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <form action = "http://localhost:5000/login"method = "post"> <p>Enter Name:</p> <p><input type = "text" name = "nm"/></p> <p><input type = "submit" value = "submit"/></p> </form> </body> </html>
loginok.html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <h1>hello {{ name }}</h1> </body> </html>
WC_public_flask.py
from flask importFlask, render_template, request app = Flask(__name__, template_folder='templates') @app.route('/')def login(): return render_template('login.html') @app.route('/login',methods = ['POST', 'GET'])def loginok(): if request.method== 'POST': user = request.form['nm'] return render_template('loginok.html',name =user) else: user = request.args.get('nm') return render_template('loginok.html',name =user) if __name__ == '__main__': app.run(debug = True)