Inside Flask - flask 扩展加载过程
Inside Flask - flask 扩展加载过程
flask 扩展(插件)通常是以 flask_<扩展名字>
为扩展的 python 包名,而使用时,可用 import flask.ext.<扩展名>
来导入扩展包。一般使用方法见 flask 扩展。在最新的 0.11.1 代码中,不建议使用 flask.ext
加载扩展,可通过 flask_xxx
的名字直接调用扩展,那样就不需要 flask
自带的扩展机制,但了解一下原来的扩展实现机制还是很有意思的。
flask 在处理这些扩展的代码在 flask/ext
和 flask/exthook.py
中。
在 ext 包的 __init__.py
中,定义了一个 setup
函数 ::
def setup():
from ..exthook import ExtensionImporter
importer = ExtensionImporter(['flask_%s', 'flaskext.%s'], __name__)
importer.install()
然后就执行 setup 函数,并删除 setup 函数。setup 函数创建一个 flask.exhook.ExtensionImporter
,这个对象是真正执行扩展包导入的地方。
ExtensionImporter使用到 python 本身的模块搜索机制。我们知道 sys.path
是 python 的模块搜索路径,使用 import 导入模块时从 sys.path
列表中逐个查找。而 sys.meta_path
从字面看是元路径,在搜索时优先于 sys.path
(或者默认的隐式模块加载器)。meta_path
里面定义其实不是路径,而是模块搜索类 finder 的列表。finder 包含 find_loader
或 find_module
方法,用于查找对应模块加载器 loader,最后调用 loader 提供的 load_module
方法加载模块(注册到 sys.modules
)。
ExtensionImporter 既是 finder 同时也是 loader。setup 函数在初始化ExtensionImporter 时,传入了一个扩展包名模式列表和当前模块的名字(flask.ext
)。 flask.ext
在这里是作为一个包装器模块(必须条件), importer 把扩展模块挂在这个模块的名字之下。['flask_%s', 'flaskext.%s']
则是搜索实际模块名时用的字符串模板,如 flask.ext.sqlalchemy
则会转换为真实模块名 flask_sqlalchemy
或 flaskext.sqlalchemy
,并尝试 import (在下文中分析)。
首先,setup 函数调用 importer 的 install 方法,让 importer 把自己注册到 sys.meta_path
中成为一个 finder ::
def install(self):
sys.meta_path[:] = [x for x in sys.meta_path if self != x] + [self]
那么,当代码中调用 from flask.ext.sqlalchemy import xxx
时,ExtensionImporter 的 find_module
函数被调用,并返回自身作为 loader ::
def find_module(self, fullname, path=None):
if fullname.startswith(self.prefix) and \
fullname != 'flask.ext.ExtDeprecationWarning':
return self
然后,按照模块导入的流程,load_module
函数被调用。
def load_module(self, fullname):
if fullname in sys.modules:
return sys.modules[fullname]
...
fullname 是模块全名 flask.ext.sqlalchemy
。在加载模块时,先从 sys.modules
查看是否已加载,如果未有加载,那么将 fullname 去掉 flask.ext
前缀,并转换为真实的模块名 ::
# 去掉前缀
modname = fullname.split('.', self.prefix_cutoff)[self.prefix_cutoff]
...
# 尝试加载真实模块
for path in self.module_choices:
realname = path % modname
try:
__import__(realname)
...
# 更新 sys.modules
module = sys.modules[fullname] = sys.modules[realname]
...
return module
至此,整个扩展模块的加载过程成功完成。
Inside Flask - flask 扩展加载过程的更多相关文章
- [转]PHP的执行流程,PHP扩展加载过程
原文:http://www.imsiren.com/archives/535 为了以后能开发PHP扩展..就一定要了解PHP的执行顺序..这篇文章就是为C开发PHP扩展做铺垫. web环境 我们假设为 ...
- Dubbo源码解析之SPI(一):扩展类的加载过程
Dubbo是一款开源的.高性能且轻量级的Java RPC框架,它提供了三大核心能力:面向接口的远程方法调用.智能容错和负载均衡,以及服务自动注册和发现. Dubbo最早是阿里公司内部的RPC框架,于 ...
- JVM——类的加载过程
附一张图方便理解,一个类的执行过程 类的加载过程,简明的来说 类装饰器就是寻找类的字节码文件并构造出类在JVM内部表示的对象组件.在Java中,类装载器把一个类装入JVM中,要经过以下步骤: 装载:查 ...
- ThinkPHP3.2 加载过程(四)
前言: 由于比较懒散,但是又是有点强迫症,所以还是想继续把ThinkPHP3.2的加载过程这个烂尾楼补充完整. ========================================分割线= ...
- Orchard 源码探索(Module,Theme,Core扩展加载概述)
参考: http://www.orchardch.com/Blog/20120830071458 1. host.Initialize(); private static IOrchardHost H ...
- Orchard Module,Theme,Core扩展加载概述
Orchard 源码探索(Module,Theme,Core扩展加载概述) 参考: http://www.orchardch.com/Blog/20120830071458 1. host.Initi ...
- Spring IOC bean加载过程
首先我们不要在学习Spring的开始产生畏难情绪.Spring没有臆想的那么高深,相反,它帮我们再项目开发中制定项目框架,简化项目开发.它的主要功能是将项目开发中繁琐的过程流程化,模式化,使用户仅在固 ...
- (转)JVM类生命周期概述:加载时机与加载过程
原文地址: http://blog.csdn.net/justloveyou_/article/details/72466105 JVM类加载机制主要包括两个问题:类加载的时机与步骤 和 类加载的方式 ...
- jvm(1)类加载(一)(加载过程,双亲加载)
JVM类加载器机制与类加载过程 jvm虚拟机的种类: Hotspot(Oracle)(基本上都是在说这个) J9, JikesRVM(IBM) Zulu, Zing (Azul) Launcher是一 ...
随机推荐
- Codeforces Round #249 (Div. 2) A - Queue on Bus Stop
水题 #include <iostream> #include <vector> #include <algorithm> using namespace std; ...
- List<string>中的泛型委托
我们先看List<T>.Sort().其定义是:public void Sort( Comparison<T> comparison ) 其要求传入的参数是Comparison ...
- jquery控制文字内容溢出用点点点(…)省略号表示
jQuery限制字符字数的方法 $(function(){ //限制字符个数 $(“.text”).each(function(){ var maxwidth=23; if($(this).text( ...
- linux pidof
转载:http://blog.51osos.com/linux/linux-pidof-command/ 什么是pidof命令? #man pidof中的解释: pidof — find the pr ...
- silverlight Canvas、StackPanel、Grid三者之间的关系
学习 silverlight 首先Canvas.StackPanel.Grid 博客园里看到jailu的这篇文章整理得很好 贴下来: Silverlight提供了非常灵活的布局管理系统,让程序员和 ...
- [LintCode] Sort Integers 整数排序
Given an integer array, sort it in ascending order. Use selection sort, bubble sort, insertion sort ...
- [转载]窗口之间的主从关系与Z-Order
窗口之间的主从关系与Z-Order 原文地址:http://www.cnblogs.com/dhatbj/p/3288152.html说明:这是本人2008年写的一篇旧文,从未公开发表过.其中除了一小 ...
- 状态压缩 DP
D - Hie with the Pie Crawling in process... Crawling failed Time Limit:2000MS Memory Limit:65536 ...
- redis虚拟机模拟集群,节点,增加多端口命令
Redis启动多端口,运行多实例 使用redis在同一台机器上,启用多个端口,实现多个实例,完成集群的模拟实现. 启动多实例 redis默认启动端口为6379,我们可以使用 --port 来指定多个端 ...
- [NoSQL]验证redis的主从复制
安装配置redis http://www.cnblogs.com/myrunning/p/4222385.html 1.1查看当前redis文件 1.2修改配置文件 拷贝配置文件分别为redis_m ...