Flask是一个使用Python编写的轻量级web框架,相比其他框架如Django更加简单易学。本文将实现一个简单示例,完成以下功能:从数据库中读取用户信息,在浏览器中分页展示,添加分页导航栏,并且实现根据用户名搜索。用到的主要知识点:数据库操作、表单、Jinja2模板。
学习环境搭建
操作系统
CentOS Linux release 7.4.1708 (Core)
执行yum update -y然后重启系统,确保软件包处于最新状态。
Python版本 2.7.5
安装Python虚拟环境
利用virtualenv可以实现一个隔离的Python环境,在这个虚拟环境中安装、删除Python的pip包而不会影响环境外的其他进程。虚拟环境还会让系统移植更容易,只要把虚拟环境的文件拷到另一个系统中然后恢复虚拟环境,就可以保证跟之前的Python包环境一模一样。
安装virtualenv:
[root@localhost ~]# pip install virtualenv
Collecting virtualenv
Downloading virtualenv-15.1.0-py2.py3-none-any.whl (1.8MB)
100% |████████████████████████████████| 1.8MB 78kB/s
Installing collected packages: virtualenv
Successfully installed virtualenv-15.1.0
创建一个名为env的虚拟环境:
[root@localhost ~]# virtualenv env
New python executable in /root/env/bin/python2
Also creating executable in /root/env/bin/python
Installing setuptools, pip, wheel...done.
进入虚拟环境env:
[root@localhost demo]# source env/bin/activate
(env) [root@localhost demo]#
前面的(env)标识了当前shell所在的虚拟环境,把env目录拷到其他系统就可以进行环境移植。
安装数据库
安装数据和相关开发包:
[root@localhost demo]# yum install mariadb mariadb-server python-devel mariadb-devel -y
安装完后,执行mysql_secure_installation,设置数据库root密码,因为是测试环境,我们为了方便设置为“test”,生产环境的话一定要设置复杂密码。
简单示例实现
项目结构
我们创建一个demo目录存放项目文件,用tree命令查看这个目录的结构如下:
(env) [root@localhost ~]# tree -I 'env' demo
demo
├── app #程序包目录
│ ├── __init__.py
│ ├── main
│ │ ├── forms.py #定义表单类
│ │ ├── __init__.py
│ │ ├── views.py #定义视图函数,实际处理http请求
│ ├── models.py #定义数据模型
│ ├── static #存放静态文件
│ └── templates #存放模板
│ ├── _macro.html
│ ├── userlist.html
├── config.py #项目配置文件
├── manage.py #项目启动脚本
└── requirements.txt #项目依赖的所有pip包
各文件的主要作用已经在上面注释。Flask并没有限定项目的结构必须如此,但是通常这是比较合理的结构,也可以根据自己的项目特点进行微调。下面分文件详细介绍如何实现本文开头提到的各个功能。
requirements.txt
这个文件保存了项目依赖的所有包及其版本,方便其他人移植你的Flask项目。文件是由命令pip freeze > requirements.txt生成的,在要运行该项目的系统中执行pip install -r requirements.txt就可以安装完所有依赖包,很方便。这个示例的requirements.txt内容如下:
alembic==0.9.6
click==6.7
Flask==0.12.2
Flask-Migrate==2.1.1
Flask-Script==2.0.6
Flask-SQLAlchemy==2.3.2
Flask-WTF==0.14.2
ForgeryPy==0.1
itsdangerous==0.24
Jinja2==2.10
Mako==1.0.7
MarkupSafe==1.0
MySQL-python==1.2.5
python-dateutil==2.6.1
python-editor==1.0.3
six==1.11.0
SQLAlchemy==1.2.0
Werkzeug==0.14.1
WTForms==2.1
config.py
项目的配置文件,集中存放配置参数,内容如下:
class Config:
SQLALCHEMY_DATABASE_URI = "mysql://root:test@localhost/demo"
SQLALCHEMY_COMMIT_ON_TEARDOWN = True
SECRET_KEY = 'hard to guess string'
@staticmethod
def init_app(app):
pass
文件中配置了数据库的访问方式,还有SECRET_KEY,这是一个密钥,用于保护表单免受跨站请求伪造(CSRF)的***。
models.py
编写业务逻辑前要先设计好数据模型,我们的示例的数据很简单,一张表——用户信息,包括两个字段:用户名和用户邮箱地址。代码如下:
from . import db
class User(db.Model):
__tablename__ = 'users'
id = db.Column(db.Integer, primary_key=True)
username = db.Column(db.String(64), unique=True, index=True)
email = db.Column(db.String(64), unique=True, index=True)
def __repr__(self):
return '
@staticmethod
def generate_fake(count=100):
from sqlalchemy.exc import IntegrityError
from random import seed
import forgery_py
seed()
for i in range(count):
u = User(username=forgery_py.internet.user_name(True),
email=forgery_py.internet.email_address())
db.session.add(u)
try:
db.session.commit()
except IntegrityError:
db.session.rollback()
Flask-SQLalchemy是Flask的数据库扩展,通过对象关系映射ORM,把对数据库的操作转换为对Python对象的操作。Class User在数据库中对应的就是一张表,表名在__tablename__中设置,两个属性username和email对应表字段。代码中还用到了一个很有意思的模块forgery_py,它用于随机生成符合要求的测试数据,函数generate_fake()随机生成数量等于count条的数据并写入数据库。生成数据的具体操作方法会在最后执行程序的时候讲。
app/__init__.py
程序包的构建文件,在这文件里定义创建程序实例的函数create_app,也叫工厂函数。工厂函数完成3个主要工作:
1. 从config.py中导入程序配置;
2. 大多数Flask的扩展需要在这个函数里初始化,例如我们的示例用到的Flask数据库扩展Flask-Sqlalchemy;
3. 还有一个工作是给程序实例注册蓝本(blueprint),蓝本用于定义URL到视图函数的路由。
代码如下:
from flask import Flask
from flask_sqlalchemy import SQLAlchemy
from config import Config
db = SQLAlchemy()
def create_app(config_name):
app = Flask(__name__)
app.config.from_object(Config)
Config.init_app(app)
db.init_app(app)
from .main import main as main_blueprint
app.register_blueprint(main_blueprint)
return app
manage.py
manage.py是整个程序的启动脚本,从上面的app/__init__.py导入工厂函数create_app(),然后创建程序实例,初始化Flask的扩展Flask-Script和Flask-Migrate,代码如下:
#!/usr/bin/env python
#conding: utf-8
import os
from flask.ext.script import Manager, Shell
from flask.ext.migrate import Migrate, MigrateCommand
from app import create_app, db
from app.models import *
app = create_app('default')
manager = Manager(app)
migrate = Migrate(app, db)
def make_shell_context():
return dict(app=app, db=db, User=User)
manager.add_command("shell", Shell(make_context=make_shell_context))
manager.add_command('db', MigrateCommand)
if __name__ == '__main__':
manager.run()
forms.py
表单的功能是接收用户提供的数据。用户将数据填到表单的各字段中,通过Post方法提交给服务端。在我们的示例中,表单用于接收用户输入的用户名,用于后续的搜索功能。代码如下:
from flask_wtf import FlaskForm
from wtforms import StringField, SubmitField
from wtforms.validators import DataRequired
class Form(FlaskForm):
name = StringField('Username', validators=[DataRequired()])
submit = SubmitField('Search')
flask.wtf是Flask框架处理表单的扩展,我们自己定义的表单Form继承该模块中的FlaskForm类。Form很简单,一共2个字段,StringField提供给用户填写要搜索的用户名,检测函数DataRequired()确保数据不为空;SubmitField是一个提交按钮,用户点击按钮后,就会把表单数据提交给服务端。
views.py
该文件中定义路由和视图函数,视图函数是实际处理http请求的函数。我们的示例只有一个视图函数,代码如下:
from flask import render_template, url_for, request, session
from . import main
from .. import db
from ..models import User
from .forms import Form
@main.route('/',methods=['GET','POST'])
def index():
page = request.args.get('page',1,type=int)
pagination = User.query.order_by(User.id).paginate(
page,per_page=10,error_out=False)
form = Form()
users = pagination.items
if form.validate_on_submit():
session['username'] = form.name.data
users = User.query.filter_by(username=session.get('username')).all()
return render_template('userlist.html',users=users,pagination=pagination,form=form)
main是一个蓝本,修饰器@main.route定义的路由的含义是,访问“/”的的GET或POST方法都由index()函数处理。开头提到了我们要实现用户信息的分页展示,所以index()先从全局变量request中获取了页数,默认显示第1页。
然后分页查询数据库,每页10个数据;
然后初始化搜索表单,If判断把视图函数的执行结果分成了表单有数据和没数据两种情况:如果表单有数据,则把表单数据存到会话session中,并且查询数据库,把符合条件的用户传递给模板;如果表单没有数据,就把所有用户传递给模板。模板根据收到的用户信息生成html页面。
视图函数最核心的任务就是返回一个html页面给客户端,render_template()通过渲染模板生成一个html页面,所谓渲染就是根据传入的参数去修改模板中对应的可变部分。模板将在下一部分详细介绍。
templates
Flask集成了Jinja2模板引擎,templates这个目录存放所有jinja2模板。模板中有一些可以被替换的部分,渲染流程也支持通过条件判断或循环控制。
_macro.html中定义了一个宏pagination_widget,宏类似于Python中的函数,这个宏用于生成分页导航栏,它接收的参数是数据库分页查找返回的对象pagination和视图函数名endpoint。pagination对象支持一系列方法和属性:
l 方法has_prev()和has_next()分别检测当前页之前和之后还有没有页,返回布尔值;
l 方法prev()和next()分别返回上一页和下一页的分页对象
l 属性page返回当前页码;
l 方法iter_pages是一个迭代器,根据参数返回一组页码;
_macro.html代码如下:
{% macro pagination_widget(pagination, endpoint) %}
li{ list-style:none;}
class="pagination">
« {% for p in pagination.iter_pages(right_current=2) %} {% if p %} {% if p == pagination.page %} href="{{ url_for(endpoint, page = p, **kwargs) }}">{{ p }} {% else %} href="{{ url_for(endpoint, page = p, **kwargs) }}">{{ p }} {% endif %} {% else %} {% endif %} {% endfor %} »
{% if not pagination.has_prev %} class="disabled"{% endif %}>
class="active">
class="disabled"> href="#">…
{% if not pagination.has_next %} class="disabled"{% endif %}>
{% endmacro %}
其核心内容是中间的for循环,根据iter_pages()方法返回的页码,生成指向该页码的URL的链接。因此分页导航栏本质是由若干链接组成的,只不过这些特殊链接都指向与自己名字相关的URL。执行效果会在本文最后一部分看到。
userlist.html是视图函数index()真正渲染的文件,它通过一个列表来展示视图函数传给它的参数users中包含的用户信息。然后它引用了上面定义的宏,生成了分页导航栏。在最后是搜索表单。
{% import "_macro.html" as macro %}
li{ list-style:none;}
用户列表:
{% for user in users%}
{{ user.username }}: {{ user.email }}
{% endfor %}
{{ macro.pagination_widget(pagination, '.index')}}
{{ form.hidden_tag() }}
{{ form.name.label }} {{ form.name() }}
{{ form.submit() }}
执行效果
生成测试数据
(env) [root@localhost demo]# python manage.py shell
manage.py:6: ExtDeprecationWarning: Importing flask.ext.script is deprecated, use flask_script instead.
from flask.ext.script import Manager, Shell
manage.py:7: ExtDeprecationWarning: Importing flask.ext.migrate is deprecated, use flask_migrate instead.
from flask.ext.migrate import Migrate, MigrateCommand
/root/demo/env/lib/python2.7/site-packages/flask_sqlalchemy/__init__.py:794: FSADeprecationWarning: SQLALCHEMY_TRACK_MODIFICATIONS adds significant overhead and will be disabled by default in the future. Set it to True or False to suppress this warning.
'SQLALCHEMY_TRACK_MODIFICATIONS adds significant overhead and '
>>> db.create_all()
>>> User.generate_fake(100)
db.create_all()创建所有表,User.generate_fake(100)生成表中的数据。
启动程序
Flask框架内置了一个用于测试的web服务器,启动服务:
(env) [root@localhost demo]# python manage.py runserver -h 0.0.0.0 -p 5000
manage.py:6: ExtDeprecationWarning: Importing flask.ext.script is deprecated, use flask_script instead.
from flask.ext.script import Manager, Shell
manage.py:7: ExtDeprecationWarning: Importing flask.ext.migrate is deprecated, use flask_migrate instead.
from flask.ext.migrate import Migrate, MigrateCommand
/root/demo/env/lib/python2.7/site-packages/flask_sqlalchemy/__init__.py:794: FSADeprecationWarning: SQLALCHEMY_TRACK_MODIFICATIONS adds significant overhead and will be disabled by default in the future. Set it to True or False to suppress this warning.
'SQLALCHEMY_TRACK_MODIFICATIONS adds significant overhead and '
* Running on http://0.0.0.0:5000/ (Press CTRL+C to quit)
访问页面
打开一个浏览器,输入URL http://x.x.x.x:5000/,x.x.x.x替换为你的测试机的ip,可以看到以下页面:
数据库中一共存了100条用户记录,分为了10页展示,点击下面的分页导航可以跳转页面,下面是第9页的内容:
然后测试搜索功能,在表单中输入“amy63”,点击按钮,结果如下: