用蓝图实现模板化应用

用蓝图实现模块化的应用

Flask用蓝图的概念来在一个应用中或跨应用制作应用组件和支持通用的模式。
蓝图很好的简化了大型应用工作的方式,并提供给Flask扩展在应用上注册操作的
核心方法。一个Blueprint对象与Flask应用对象的工作方式很像,但它确实不是
一个应用,而是一个描述如何构建或扩展应用的蓝图。

为什么使用蓝图
Flask中的蓝图为这些情况设计:
1 - 把一个应用分解为一个蓝图的集合。这对大型应用是理想的。一个项目可以
实例化一个应用对象,初始化几个扩展,并注册一集合的蓝图。
2 - 以URL前缀和/或子域名,在应用上注册一个蓝图。URL前缀/子域名中的参数
即成为这个蓝图下所有视图函数的共同的视图函数(默认情况下)。
3 - 在一个蓝图中用不同的URL规则多次注册一个蓝图。
4 - 通过蓝图提供模板过滤器,静态文件,模板和其他功能。一个蓝图不一定要
实现应用或者视图函数。
5 - 初始化一个FLask扩展时,在这些情况中注册一个蓝图。

Flask中的蓝图不是即插应用,因为它实际上并不是一个应用———它是可以注册,
甚至可以多次注册到应用山的操作集合。为什么不适用多个应用对象?可以实现
,但是你的应用的配置是分开的,并在WSGI层管理。
蓝图作为Flask层提供分割的替代,共享应用配置,并且在必要情况下可以更改所
注册的应用对象。它的缺点是你不能在应用创建后撤销注册一个蓝图而不销毁整个
应用对象。

蓝图的设想
蓝图的基本设想是当它们注册到应用上时,它们记录将会被执行的操作。当分派
请求和生成从一个端点到另一个的URL时,Flask会关联蓝图中的视图函数。

第一个蓝图
from flask import Blueprint,render_template,abort
from jinja2 import TemplateNotFound

simple_page = Blueprint('simple_page',__name__,template_folder='templates')

@simple_page.route('/',defaults={'page':'index'})
@simple_page.route('/<page>')
def show(page):
	try:
		return render_template('pages/%s' % page)
	except TemplateNotFound:
		abort(404)
当我们使用@simple_page.route装饰器绑定函数时,在蓝图之后被注册时它会记录
把show函数注册到应用上的意图。此外,它会给函数的端点加上由Blueprint的
构造函数中给出的蓝图的名称作为前缀(在此例中时simple_page)

注册蓝图
from flask import Flask
from yourapplication.simple_page import simple_page
app = Flask(__name__)
app.register_blueprint(simple_page)
如果检查已经注册到应用的规则,就会发现
[<Rule '/static/<filename>' (HEAD, OPTIONS, GET) -> static>,
 <Rule '/<page>' (HEAD, OPTIONS, GET) -> simple_page.show>,
 <Rule '/' (HEAD, OPTIONS, GET) -> simple_page.show>]
 第一个是来自应用自身,用于静态文件。其他的两个用于simple_page蓝图中的
 show函数。它们的前缀是蓝图的名称,并且用一个点(.)来分割。
不过蓝图也可以在不同的位置挂载:
app.register_blueprint(simple_page,url_prefix='/pages')
那么生成的规则如下
[<Rule '/static/<filename>' (HEAD, OPTIONS, GET) -> static>,
 <Rule '/pages/<page>' (HEAD, OPTIONS, GET) -> simple_page.show>,
 <Rule '/pages/' (HEAD, OPTIONS, GET) -> simple_page.show>]
 在此之上,可以多次注册蓝图,虽然不是每个蓝图都会正确的响应这些。实际上,
 蓝图能否被多次挂载,取决于蓝图是怎样实现的。

 蓝图资源
 蓝图也可以提供资源。有时候你会只为它提供的资源而引入一个蓝图

 蓝图资源文件夹
 像常规的应用一样,蓝图被设想为包含在一个文件夹中。当多个蓝图源于同一个
 文件夹时,可以不必考虑上述情况,但这也通常是不推荐的做法。
 这个文件夹会从Blueprint的第二个参数中推断出来,通常是__name__。这个参数
 决定对应蓝图的时哪个逻辑的Python模块或包。如果它指向一个存在的Python包,
 这个包(通常是文件系统中的文件夹)就是资源文件夹。如果是一个模块,模块
 所在的包就是资源文件夹。可以访问Blueprint.root_path属性来查看资源文件夹
 是什么
 >>> simple_page.root_path
'/Users/username/TestProject/yourapplication'
可以使用open_resource()函数来快速从这个文件夹打开源文件:
with simple_page.open_resource('static/style.css') as f:
	code = f.read()

静态文件
一个蓝图可以通过static_folder关键字参数提供一个指向文件系统上文件夹的路径,
来暴露一个带有静态文件的文件夹。这可以是一个绝对路径,也可以是相对于蓝图
文件夹的路径:
admin = Blueprint('admin',__name__,static_folder='static')
默认情况下,路径最右边的部分就是它在web上所暴露的地址。因为这里这个文件夹
叫做static,它会在蓝图+/static的位置上可用。也就是说,蓝图为/admin把静态
文件夹注册到/admin/static
最后是命名的blueprint_name.static,这样你可以生成它的URL,就像对应用的静态
文件夹所做的那样:
url_for('admin.static',filename='style.css')

模板和其他功能
模板
如果你想要蓝图暴露模板,你可以提供 Blueprint 构造函数中的 template_folder 参数来实现:

admin = Blueprint('admin', __name__, template_folder='templates')
像对待静态文件一样,路径可以是绝对的或是相对蓝图资源文件夹的。模板文件夹会被加入到模板
的搜索路径中,但是比实际的应用模板文件夹优先级低。这样,你可以容易地在实际的应用中
覆盖蓝图提供的模板。

那么当你有一个 yourapplication/admin 文件夹中的蓝图并且你想要渲染 'admin/index.html' 模板,
且你已经提供了 templates 作为 template_folder ,
你需要这样创建文件: yourapplication/admin/templates/admin/index.html

构造 URL
当你想要从一个页面链接到另一个页面,你可以像通常一个样使用 url_for() 函数,
只是你要在 URL 的末端加上蓝图的名称和一个点( . )作为前缀:

url_for('admin.index')
此外,如果你在一个蓝图的视图函数或是模板中想要从链接到同一蓝图下另一个端点,
 你可以通过对端点只加上一个点作为前缀来使用相对的重定向:

url_for('.index')
这个案例中,它实际上链接到 admin.index ,假如请求被分派到任何其它的 admin 蓝图端点。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值