位置: 编程技术 - 正文

使用Python的Flask框架构建大型Web应用程序的结构示例(flask pycharm)

编辑:rootadmin

推荐整理分享使用Python的Flask框架构建大型Web应用程序的结构示例(flask pycharm),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:python中flask干什么用,flask调用python程序,python中flask干什么用,python flask,python flask快速入门与进阶,python flask django,python+flask,python-flask,内容如对您有帮助,希望把文章链接给更多的朋友!

虽然小型web应用程序用单个脚本可以很方便,但这种方法却不能很好地扩展。随着应用变得复杂,在单个大的源文件中处理会变得问题重重。

与大多数其他web框架不同,Flask对大型项目没有特定的组织方式;应用程序的结构完全交给开发人员自己决定。在这一章,提出一个可能的方式来组织管理一个大型应用程序的包和模块。这种结构将用于书中其余的示例中。

1、项目结构

示例 基本多文件Flask应用结构

这个结构有四个顶层目录:

Flask应用一般放置在名为app的目录下。 migrations目录包含数据库迁移脚本,这和之前说的一样。 单元测试放置在test目录下 venv目录包含Python虚拟环境,这和之前说的也是一样的。

还有一些新的文件:

requirements.txt列出一些依赖包,这样就可以很容易的在不同的计算机上部署一个相同的虚拟环境。 config.py存储了一些配置设置。 manage.py用于启动应用程序和其他应用程序任务。

为了帮助你完全理解这个结构,下面会描述将hello.py应用改为符合这一结构的整个流程。

2、配置选项应用程序通常需要几个配置设置。最好的例子就是在开发过程中需要使用不同的数据库,测试,生产环境,这样他们可以做到互不干扰。

我们可以使用配置类的层次结构来代替hello.py中的简单类字典结构配置。下面展示了config.py文件。

config.py:应用程序配置

Config基类包含一些相同配置;不同的子类定义不同的配置。额外配置可以在需要的时候在加入。

为了让配置更灵活更安全,一些设置可以从环境变量中导入。例如,SECRET_KEY,由于它的敏感性,可以在环境中设置,但如果环境中没有定义就必须提供一个默认值。

在三个配置中SQLALCHEMY_DATABASE_URI变量可以分配不同的值。这样应用程序可以在不同的配置下运行,每个可以使用不同的数据库。

配置类可以定义一个将应用程序实例作为参数的init_app()静态方法。这里特定于配置的初始化是可以执行的。这里Config基类实现一个空init_app()方法。

在配置脚本的底部,这些不同的配置是注册在配置字典中。将其中一个配置(开发配置)注册为默认配置。

3、应用程序包应用程序包放置了所有应用程序代码、模板和静态文件。它被简单的称为app,也可以给定一个特定于应用的名称(如果需要的话)。templates和static目录是应用的一部分,因此这两个目录应该放置在app中。数据库模型和电子邮件支持功能也要置入到这个包中,每个都以app/models.py和app/email.py形式存入自己的模块当中。

3.1、使用一个应用程序工厂

在单个文件中创建应用程序的方式非常方便,但是它有一个大缺点。因为应用程序创建在全局范围,没有办法动态的适应应用配置的更改:脚本运行时,应用程序实例已经创建,所以它已经来不及更改配置。对于单元测试这是特别重要的,因为有时需要在不同的配置下运行应用程序来获得更好的测试覆盖率。

解决这一问题的方法就是将应用程序放入一个工厂函数中来延迟创建,这样就可以从脚本中显式的调用。

这不仅给脚本充足的时间来设置配置,也能用于创建多个应用程序实例——一些在测试过程中非常有用的东西。被定义在app包的构造函数中的应用程序工厂函数会在示例7-3中展示。

这个构造函数导入大部分当前需要使用的扩展,但因为没有应用程序实例初始化它们,它可以被创建但不初始化通过不传递参数给它们的构造函数。create_app()即应用程序工厂函数,需要传入用于应用程序的配置名。配置中的设置被保存在config.py中的一个类中,可以使用Flask的app.config配置对象的from_object()方法来直接导入。配置对象可以通过对象名从config字典中选出。一旦应用程序被创建且配置好,扩展就可以被初始化。调用扩展里的init_app()之前先创建并完成初始化工作。

app/ _init__.py:应用程序包构造函数_

工厂函数返回创建的应用程序实例,但是请注意,在当前状态下使用工厂函数创建的应用程序是不完整的,因为它们没有路由和自定义错误页面处理程序。这是下一节的主题。

3.2、在蓝图中实现应用程序的功能

应用程序工厂的转化工作引出了路由的复杂化。在单脚本应用中,应用程序实例是全局的,所以可以很容易地使用app.route装饰器定义路由。但是现在应用程序在运行时创建,app.route装饰器只有在create_app()调用后才开始存在,这就太迟了。就像路由那样,这些通过app.errorhandler装饰器定义的自定义错误页面处理程序也存在同样的问题。

幸运的是Flask使用蓝图来提供一个更好的解决方案。一个蓝图就类似于一个可以定义路由的应用程序。不同的是,和路由相关联的蓝图都在休眠状态,只有当蓝图在应用中被注册后,此时的路由才会成为它的一部分。使用定义在全局作用域下的蓝图,定义应用程序的路由就几乎可以和单脚本应用程序一样简单了。

和应用程序一样,蓝图可以定义在一个文件或一个包中与多个模块一起创建更结构化的方式。为了追求最大的灵活性,可以在应用程序包中创建子包来持有蓝图。下面展示了创建蓝图的构造函数。

app/main/ _init__.py:创建蓝图_

蓝图是通过实例化Blueprint类对象来创建的。这个类的构造函数接收两个参数:蓝图名和蓝图所在的模块或包的位置。与应用程序一样,在大多数情况下,对于第二个参数值使用Python的__name__变量是正确的。

应用程序的路由都保存在app/main/views.py模块内部,而错误处理程序则保存在app/main/errors.py中。导入这些模块可以使路由、错误处理与蓝图相关联。重要的是要注意,在app/init.py脚本的底部导入模块要避免循环依赖,因为view.py和errors.py都需要导入main蓝图。

蓝图和应用程序一样注册在create_app()工厂函数中,如下所示。

使用Python的Flask框架构建大型Web应用程序的结构示例(flask pycharm)

示例 app/ _init__.py:蓝图注册_

下面则展示了错误处理。

app/main/errors.py:蓝图的错误处理

在蓝图中写错误处理的不同之处是,如果使用了errorhandler装饰器,则只会调用在蓝图中引起的错误处理。而应用程序范围内的错误处理则必须使用app_errorhandler。

这里展示了被更新在蓝图中的应用程序路由。

app/main/views.py:带有蓝图的应用程序路由

在蓝图中写视图函数有两大不同点。第一,正如之前的错误处理一样,路由装饰器来自于蓝图。第二个不同是url_for()函数的使用。你可能会回想,该函数的第一个参数为路由节点名,它给基于应用程序的路由指定默认视图函数。例如,单脚本应用程序中的index()视图函数的URL可以通过url_for('index')来获得。

不同的是Flask名称空间适用于来自蓝图的所有节点,这样多个蓝图可以使用相同节点定义视图函数而不会产生冲突。名称空间就是蓝图名(Blueprint构造函数中的第一个参数),所以index()视图函数注册为main.index且它的URL可以通过url_for('main.index')获得。

在蓝图中,url_for()函数同样支持更短格式的节点,省略蓝图名,例如url_for('.index')。有了这个,就可以这样使用当前请求的蓝图了。这实际意味着相同蓝图内的重定向可以使用更短的形式,如果重定向跨蓝图则必须使用带名称空间的节点名。

完成了应用程序页面更改,表单对象也保存在app/main/forms.py模块中的蓝图里面。

4、启动脚本顶层目录中的manage.py文件用于启动应用。

manage.py:启动脚本

这个脚本开始于创建应用程序。使用环境变量FLASK_CONFIG,若它已经定义了则从中获取配置;如果没有,则是用默认配置。然后用于Python shell的Flask-Script、Flask-Migrate以及自定义上下文会被初始化。

为了方便,会增加一行执行环境,这样在基于Unix的操作系统上可以通过./manage.py来执行脚本来替代冗长的python manage.py。

5、需求文件应用程序必须包含requirements.txt文件来记录所有依赖包,包括精确的版本号。这很重要,因为可以在不同的机器上重新生成虚拟环境,例如在生产环境的机器上部署应用程序。这个文件可以通过下面的pip命令自动生成:

当安装或更新一个包之后最好再更新一下这个文件。以下展示了一个需求文件示例:

当你需要完美复制一个虚拟环境的时候,你可以运行以下命令创建一个新的虚拟环境:

当你读到这时,示例requirements.txt文件中的版本号可能已经过时了。如果喜欢你可以尝试用最近发布的包。如果遇到任何问题,你可以随时回退到需求文件中与应用兼容的指定版本。

6、单元测试这个应用非常小以至于不需要太多的测试,但是作为示例会在示例中展示两个简单的测试定义。

示例:tests/test_basics.py:单元测试

编写好的测试使用的是来自于Python标准库中标准的unittest包。setUp()和tearDown()方法在每个测试之前和之后运行,且任何一个方法必须以test_开头作为测试来执行。

建议:如果你想要学习更多使用Python的unittest包来写单元测试的内容,请参阅官方文档。setUp()方法尝试创建一个测试环境,类似于运行应用程序。首先它创建应用程序配置用于测试并激活上下文。这一步确保测试可以和常规请求一样访问current_app。然后,当需要的时候,可以创建一个供测试使用的全新数据库。数据库和应用程序上下文会在tearDown()方法中被移除。

第一个测试确保应用程序实例存在。第二个测试确保应用程序在测试配置下运行。为了确保tests目录有效,需要在tests目录下增加__init__.py文件,不过该文件可以为空,这样unittest包可以扫描所有模块并定位测试。

建议:如果你有克隆在GitHub上的应用程序,你现在可以运行git checkout 7a来切换到这个版本的应用程序。为了确保你已经安装了所有依赖集,需要运行pip install -r requirements.txt。为了运行单元测试,可以在manage.py脚本中增加一个自定义的命令。

下面展示如何添加测试命令。

示例:manage.pyt:单元测试启动脚本

manager.command装饰器使得它可以很容易的实现自定义命令。被装饰的函数名可以被当做命令名使用,且函数的文档字符串会显示帮助信息。test()函数的执行会调用unittest包中的测试运行器。

单元测试可以像下面这样执行:

7、数据库启动与单脚本的应用相比,重构后的应用使用不同数据库。

从环境变量中获取的数据库URL作为首选,默认SQLite数据库作为可选。三个配置中的环境变量和SQLite数据库文件名是不一样的。例如,开发配置的URL是从DEV_DATABASE_URL环境变量中获取,如果没有定义则会使用名为data-dev.sqlite的SQLite数据库。

无论数据库URL源的是哪一个,都必须为新的数据库创建数据库表。如果使用了Flask-Migrate来保持迁移跟踪,数据库表可以被创建或更新到最近的版本通过下面的命令:

相信与否,已经到了第一部分结束的地方。你现在已经学到了Flask必要的基本要素,但是你不确定如何将这些零散的知识组合在一起形成一个真正的应用程序。第二部分的目的是通过开发一个完整的应用程序来带领你继续前行。

浅析AST抽象语法树及Python代码实现 在计算机科学中,抽象语法树(abstractsyntaxtree或者缩写为AST),或者语法树(syntaxtree),是源代码的抽象语法结构的树状表现形式,这里特指编程语言

python下调用pytesseract识别某网站验证码的实现方法 一、pytesseract介绍1、pytesseract说明pytesseract最新版本0.1.6,网址:

深入理解python函数递归和生成器 一、什么是递归如果函数包含了对其自身的调用,该函数就是递归的。递归做为一种算法在程序设计语言中广泛应用,它通常把一个大型复杂的问题层

标签: flask pycharm

本文链接地址:https://www.jiuchutong.com/biancheng/387574.html 转载请保留说明!

上一篇:在Python的Flask框架中构建Web表单的教程(python框架flask系列)

下一篇:浅析AST抽象语法树及Python代码实现(asoul抽象)

  • 小规模进项可以收专票吗
  • 有限合伙企业应当
  • 被客户扣钱怎么办
  • 差额征税所得税怎么申报
  • 事业单位缴纳增值税标准
  • 资产负债表的应付职工薪酬怎么填
  • 贴现法付息什么意思
  • 生产企业电梯维修方案
  • 固定资产清理不能有余额
  • 转账支票需要哪些资料
  • 年金计算个税需要扣除吗
  • 小规模纳税人超过500万可以不转一般纳税人吗
  • 公司为员工需要承担哪些责任
  • 零申报的企业
  • 全员劳动生产率是什么意思
  • 如何查发票真伪发票查询
  • 同业清算互联前置 骗局
  • 投资收益年底结转怎么算
  • 薪酬费用属于什么科目
  • 高速公路通行费抵扣最新规定
  • 个人租房给公司开票税点
  • 物流公司怎么进去工作的
  • 鸿蒙系统智能设备怎么开启
  • 公司的固定资产是什么
  • 如何让游戏速度加快
  • 代办营业执照费用税务编码是多少
  • 安全生产责任险保障范围
  • 核定征收的企业利润怎么处理
  • PHP:stream_set_chunk_size()的用法_Stream函数
  • 土地增值税安置房收入的确认原则
  • 购货方开具红字发票怎么做账
  • php 访问数据库
  • 小规模企业需要交几个点
  • 把ChatGPT接入我的个人网站
  • 出售债券税费处理会计分录怎么写
  • 毕业设计烦死了
  • promise的几种状态
  • 企业计提养老保险分录
  • 买一赠一的销售方式
  • 帝国cms到底好不好
  • sql2008安装出现以下错误
  • mongodb起源
  • 免税不能抵扣
  • 建造合同收入的会计处理
  • 公司注销了,账务是不是可以销毁了
  • 应收账款账龄计提坏账比例
  • 其他应收款财务报表取数
  • 投资性房地产累计摊销
  • 其他应付款的辅助科目是什么
  • 车间报销维修费会计科目
  • 周转材料怎么做分录
  • 应收票据的账务处理程序
  • 预发绩效会计分录
  • 加计抵减期末有余额怎么办
  • 损益类账户借贷方向增减
  • 人工费用和管理费用比例
  • 企业成立第二年有补贴吗
  • 记账凭证的填制与审核
  • solaris syslog配置
  • 苹果mac怎么复制文字
  • 服务器centos版本选择
  • 进程crash是什么意思
  • ubuntu系统安装程序
  • 远程电脑屏幕桌面图标什么都没有了
  • 新版电脑装win7
  • [置顶] 《精神怪谈》 后续起点
  • 批处理在windows中的典型应用
  • perl 数组放入另一个数组
  • 被调用的对象已与其客户端断开连接怎么办
  • 如何大小写字母转换
  • php redis incr
  • ztree重新加载数据
  • shell读取文本
  • python抢红包
  • jQuery插件能输出到控制台
  • jquery禁用输入框
  • js function的this指向
  • 江苏国税电子税局
  • 地方税务局怎么报税
  • 车船税代收有发票吗
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设