Flask之测试与部署
5.1 蓝图Blueprint
为什么学习蓝图?
我们学习Flask框架,是从写单个文件,执行hello world开始的。我们在这单个文件中可以定义路由、视图函数、定义模型等等。但这显然存在一个问题:随着业务代码的增加,将所有代码都放在单个程序文件中,是非常不合适的。这不仅会让代码阅读变得困难,而且会给后期维护带来麻烦。
如下示例:我们在一个文件中写入多个路由,这会使代码维护变得困难。
- from flask import Flask
- app = Flask(__name__)
- @app.route('/')
- def index():
- return 'index'
- @app.route('/list')
- def list():
- return 'list'
- @app.route('/detail')
- def detail():
- return 'detail'
- @app.route('/')
- def admin_home():
- return 'admin_home'
- @app.route('/new')
- def new():
- return 'new'
- @app.route('/edit')
- def edit():
- return 'edit'
问题:一个程序执行文件中,功能代码过多。就是让代码模块化。根据具体不同功能模块的实现,划分成不同的分类,降低各功能模块之间的耦合度。python中的模块制作和导入就是基于实现功能模块的封装的需求。
尝试用模块导入的方式解决: 我们把上述一个py文件的多个路由视图函数给拆成两个文件:app.py和admin.py文件。app.py文件作为程序启动文件,因为admin文件没有应用程序实例app,在admin文件中要使用app.route路由装饰器,需要把app.py文件的app导入到admin.py文件中。
- # 文件app.py
- from flask import Flask
- # 导入admin中的内容
- from admin import *
- app = Flask(__name__)
- @app.route('/')
- def index():
- return 'index'
- @app.route('/list')
- def list():
- return 'list'
- @app.route('/detail')
- def detail():
- return 'detail'
- if __name__ == '__main__':
- app.run()
- # 文件admin.py
- from app import app
- @app.route('/')
- def admin_home():
- return 'admin_home'
- @app.route('/new')
- def new():
- return 'new'
- @app.route('/edit')
- def edit():
- return 'edit'
启动app.py文件后,我们发现admin.py文件中的路由都无法访问。 也就是说,python中的模块化虽然能把代码给拆分开,但不能解决路由映射的问题。
什么是蓝图?
蓝图:用于实现单个应用的视图、模板、静态文件的集合。
蓝图就是模块化处理的类。
简单来说,蓝图就是一个存储操作路由映射方法的容器,主要用来实现客户端请求和URL相互关联的功能。 在Flask中,使用蓝图可以帮助我们实现模块化应用的功能。
蓝图的运行机制:
蓝图是保存了一组将来可以在应用对象上执行的操作。注册路由就是一种操作,当在程序实例上调用route装饰器注册路由时,这个操作将修改对象的url_map路由映射列表。当我们在蓝图对象上调用route装饰器注册路由时,它只是在内部的一个延迟操作记录列表defered_functions中添加了一个项。当执行应用对象的 register_blueprint() 方法时,应用对象从蓝图对象的 defered_functions 列表中取出每一项,即调用应用对象的 add_url_rule() 方法,这将会修改程序实例的路由映射列表。
蓝图的使用:
一、创建蓝图对象。
- #Blueprint必须指定两个参数,admin表示蓝图的名称,__name__表示蓝图所在模块
- admin = Blueprint('admin',__name__)
二、注册蓝图路由。
- @admin.route('/')
- def admin_index():
- return 'admin_index'
三、在程序实例中注册该蓝图。
- app.register_blueprint(admin,url_prefix='/admin')
文件目录:
程序执行文件/test4/test.py
- from flask import Flask
- #导入蓝图对象
- from login import logins
- from user import users
- app = Flask(__name__)
- @app.route('/')
- def hello_world():
- return 'Hello World!'
- #注册蓝图,第一个参数logins是蓝图对象,url_prefix参数默认值是根路由,如果指定,会在蓝图注册的路由url中添加前缀。
- app.register_blueprint(logins,url_prefix='')
- app.register_blueprint(users,url_prefix='')
- if __name__ == '__main__':
- app.run(debug=True)
创建蓝图:/test4/user.py
- from flask import Blueprint,render_template
- #创建蓝图,第一个参数指定了蓝图的名字。
- users = Blueprint('user',__name__)
- @users.route('/user')
- def user():
- return render_template('user.html')
创建蓝图:/test4/login.py
- from flask import Blueprint,render_template
- #创建蓝图
- logins = Blueprint('login',__name__)
- @logins.route('/login')
- def login():
- return render_template('login.html')
运行/test4/test.py文件
动态路由示例(作者--图书):
文件目录:Flask_test4/delete.py
- from flask import Blueprint,redirect,url_for
- app_au = Blueprint('app_au',__name__)
- app_bk = Blueprint('app_bk',__name__)
- from test4 import *
- @app_au.route('/delete_au<id>')
- def delete_au(id):
- del_au = Author.query.filter_by(id=id).first()
- db.session.delete(del_au)
- db.session.commit()
- return redirect(url_for('index'))
- @app_bk.route('/delete_bk<id>')
- def delete_bk(id):
- del_bk = Book.query.filter_by(id=id).first()
- db.session.delete(del_bk)
- db.session.commit()
- return redirect(url_for('index'))
文件目录:Flask_test4/test4.py
- #coding=utf-8
- #目的:创建两个模型类型,实现数据库的连接和数据的操作
- from flask import Flask,render_template,request,redirect,url_for
- from flask_sqlalchemy import SQLAlchemy
- from flask_wtf import FlaskForm
- from wtforms import StringField,SubmitField
- from wtforms.validators import DataRequired
- #导入delete文件中的蓝图对象
- from delete import app_au,app_bk
- app = Flask(__name__)
- #对数据库连接的基本设置
- app.config['SQLALCHEMY_DATABASE_URI']='mysql://root:mysql@localhost/test0'
- app.config['SQLALCHEMY_COMMIT_ON_TEARDOWN'] = True
- app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = True
- #把应用程序的实例和SQLAlchemy进行关联
- db = SQLAlchemy(app)
- app.config['SECRET_KEY'] = 'a'
- #自定义表单,实现数据的输入保存操作
- class Append(FlaskForm):
- author = StringField(validators=[DataRequired()])
- book = StringField(validators=[DataRequired()])
- submit = SubmitField(u'提交')
- #自定义模型类
- class Author(db.Model):
- __tablename__ = 'authors'
- id = db.Column(db.Integer,primary_key=True)
- name = db.Column(db.String(32),unique=True)
- def __repr__(self):
- return 'author:%s'%self.name
- class Book(db.Model):
- __tablename__ = 'books'
- id = db.Column(db.Integer,primary_key=True)
- info = db.Column(db.String(32),unique=True)
- def __repr__(self):
- return 'book:%s'%self.info
- @app.route('/',methods=['GET','POST'])
- def index():
- au = Author.query.all()
- bk = Book.query.all()
- form = Append()
- if form.validate_on_submit():
- #从表单中获取数据
- wtf_au = form.author.data
- wtf_bk = form.book.data
- #把数据存入模型类中
- db_au = Author(name=wtf_au)
- db_bk = Book(info=wtf_bk)
- #添加到数据库操作
- db.session.add_all([db_au,db_bk])
- db.session.commit()
- au = Author.query.all()
- bk = Book.query.all()
- return render_template('index.html',au=au,bk=bk,form=form)
- if request.method == 'GET':
- return render_template('index.html',au=au,bk=bk,form=form)
- #注册蓝图
- app.register_blueprint(app_au)
- app.register_blueprint(app_bk)
- if __name__ == '__main__':
- print app.url_map
- app.run(debug=True)
查看蓝图路由:蓝图路由可以分为两块,"."前面的是蓝图名称,"."后面的是视图函数名。
Flask之测试与部署的更多相关文章
- 用Jenkins+Gradle+Jetty实现持续集成、测试、部署
自动集成有很多种方案,本例用到的工具是Jenkins(前身Hudson)+Gradle+Jetty,关于Gradle可参考上一篇,Gradle常见问题. 本例项目名称: WAP Jetty 安装Jen ...
- 使用Bitbucket Pipeline进行.Net Core项目的自动构建、测试和部署
1. 引言 首先,Bitbucket提供支持Mercurial和Git版本控制系统的网络托管服务.简单来说,它类似于GitHub,不同之处在于它支持个人免费创建私有项目仓库.除此之外,Bitbucke ...
- 基于flask+gunicorn+nginx来部署web App
基于flask+gunicorn&&nginx来部署web App WSGI协议 Web框架致力于如何生成HTML代码,而Web服务器用于处理和响应HTTP请求.Web框架和Web服务 ...
- Spring Boot(十二):spring boot如何测试打包部署
Spring Boot(十二):spring boot如何测试打包部署 一.开发阶段 1,单元测试 在开发阶段的时候最重要的是单元测试了,springboot对单元测试的支持已经很完善了. (1)在p ...
- Maven创建Web工程并执行构建/测试/打包/部署
创建工程基本参考上一篇Java Application工程,不同的是命令参数变了,创建Web工程的命令如下: mvn archetype:generate -DgroupId=com.jsoft.te ...
- 使用Jenkins结合Gogs和SonarQube对项目代码进行测试、部署、回滚,以及使用keepalived+haproxy调度至后端tomcat
0 环境说明 主tomcat:192.168.0.112 备tomcat:192.168.0.183 haproxy+keepalived-1:192.168.0.156 haproxy+keepal ...
- 【新书推荐】《ASP.NET Core微服务实战:在云环境中开发、测试和部署跨平台服务》 带你走近微服务开发
<ASP.NET Core 微服务实战>译者序:https://blog.jijiechen.com/post/aspnetcore-microservices-preface-by-tr ...
- 使用 Visual Studio 开发、测试和部署 Azure Functions(二)测试,部署
1,引言 上一篇介绍了使用使用 Visual Studio 开发 "Azure Functions" 函数,此篇介绍 “Azure Functions” 的测试以及直接从 Vist ...
- python web开发c6——阿里云上ubuntu+flask+gunicorn+nginx服务器部署(一)简单测试
简述 Nginx在服务器部署中的作用 请求通过Nginx实现反向代理,将请求提交给代理服务器.本文中只用了一台服务器,所以是代理到本机. gunicorn的作用 作为服务器代码的容器.接收Nginx的 ...
随机推荐
- cassandra框架模型之二——存储机制 CommitLog MemTable SSTable
四.副本存储 Cassandra不像HBase是基于HDFS的分布式存储,它的数据是存在每个节点的本地文件系统中. Cassandra有三种副本配置策略: 1) SimpleStrategy (Rac ...
- 【poj2155】Matrix(二维树状数组区间更新+单点查询)
Description Given an N*N matrix A, whose elements are either 0 or 1. A[i, j] means the number in the ...
- EditPlus保存时不生成bak文件(转)
如何设置EditPlus保存时不生成bak文件 EditPlus是一个强大的编辑工具,不单单是编辑文字强大,很多的刚开始学习编程语言的初学者会选择它,例如html,js,php,java.小编刚开始学 ...
- Java学习笔记——基础篇
Tips1:eclipse中会经常用到System.out.println方法,可以先输入syso,然后eclipse就会自动联想出这个语句了!! 学习笔记: *包.权限控制 1.包(package) ...
- node.js之npm库
npm库安装可分为本地安装和全局安装,本地安装如下: npm install <Module name> 但是好像会出问题,官方貌似推荐全局安装,so 全局安装官方所给命令如下: npm ...
- Friendly ARM linux交叉编译问题解决
ARM-LINUX-GCC 安装参考:(笔记)Ubuntu下安装arm-linux-gcc-4.4.3.tar.gz (交叉编译环境) 然而安装完成之后运行 arm-linux-gcc -v (注意g ...
- CSS3实现多列纵向滚动
效果如图: 小程序wxml: <view class='wraper'> <view class="header"> 头部 </view> &l ...
- 剑指offer-第四章解决面试题思路之总结
- noip济南清北冲刺班DAY1
上午 T1 立方数 题目描述 LYK定义了一个数叫“立方数”,若一个数可以被写作是一个正整数的3次方,则这个数就是立方数,例如1,8,27就是最小的3个立方数. 现在给定一个数P,LYK想要知道这个数 ...
- Xcode 打开playground文件的时候提示-Unable to find execution service for selected run destination
解决办法: step 1: 关闭Xcode (快捷键cmd + q) step 2:在terminal里面运行如下语句 rm -rf ~/Library/Developer/CoreSimulator ...