Inside Flask - flask 扩展加载过程

flask 扩展(插件)通常是以 flask_<扩展名字> 为扩展的 python 包名,而使用时,可用 import flask.ext.<扩展名> 来导入扩展包。一般使用方法见 flask 扩展。在最新的 0.11.1 代码中,不建议使用 flask.ext 加载扩展,可通过 flask_xxx 的名字直接调用扩展,那样就不需要 flask 自带的扩展机制,但了解一下原来的扩展实现机制还是很有意思的。

flask 在处理这些扩展的代码在 flask/extflask/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_loaderfind_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_sqlalchemyflaskext.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 扩展加载过程的更多相关文章

  1. [转]PHP的执行流程,PHP扩展加载过程

    原文:http://www.imsiren.com/archives/535 为了以后能开发PHP扩展..就一定要了解PHP的执行顺序..这篇文章就是为C开发PHP扩展做铺垫. web环境 我们假设为 ...

  2. Dubbo源码解析之SPI(一):扩展类的加载过程

    Dubbo是一款开源的.高性能且轻量级的Java RPC框架,它提供了三大核心能力:面向接口的远程方法调用.智能容错和负载均衡,以及服务自动注册和发现. Dubbo最早是阿里公司内部的RPC框架,于 ...

  3. JVM——类的加载过程

    附一张图方便理解,一个类的执行过程 类的加载过程,简明的来说 类装饰器就是寻找类的字节码文件并构造出类在JVM内部表示的对象组件.在Java中,类装载器把一个类装入JVM中,要经过以下步骤: 装载:查 ...

  4. ThinkPHP3.2 加载过程(四)

    前言: 由于比较懒散,但是又是有点强迫症,所以还是想继续把ThinkPHP3.2的加载过程这个烂尾楼补充完整. ========================================分割线= ...

  5. Orchard 源码探索(Module,Theme,Core扩展加载概述)

    参考: http://www.orchardch.com/Blog/20120830071458 1. host.Initialize(); private static IOrchardHost H ...

  6. Orchard Module,Theme,Core扩展加载概述

    Orchard 源码探索(Module,Theme,Core扩展加载概述) 参考: http://www.orchardch.com/Blog/20120830071458 1. host.Initi ...

  7. Spring IOC bean加载过程

    首先我们不要在学习Spring的开始产生畏难情绪.Spring没有臆想的那么高深,相反,它帮我们再项目开发中制定项目框架,简化项目开发.它的主要功能是将项目开发中繁琐的过程流程化,模式化,使用户仅在固 ...

  8. (转)JVM类生命周期概述:加载时机与加载过程

    原文地址: http://blog.csdn.net/justloveyou_/article/details/72466105 JVM类加载机制主要包括两个问题:类加载的时机与步骤 和 类加载的方式 ...

  9. jvm(1)类加载(一)(加载过程,双亲加载)

    JVM类加载器机制与类加载过程 jvm虚拟机的种类: Hotspot(Oracle)(基本上都是在说这个) J9, JikesRVM(IBM) Zulu, Zing (Azul) Launcher是一 ...

随机推荐

  1. Hadoop维护IPC链接

    IPC链接上长时间没有发生远程调用,客户端会发送一个心跳消息给服务器端,用于维护链接. Connection的lastActivity用来记录上次发生IPC通信的时间. Connection.touc ...

  2. 编写unit test以及自动化测试WebDriver

    http://msdn.microsoft.com/en-us/library/hh694602.aspx#BKMK_Quick_starts   http://www.seleniumhq.org/ ...

  3. 如何在 ejs 模板中使用 helper function 外部函数进行特殊处理?

    一般我们想要在 ejs 模板中使用外部函数用于特殊的处理,比如:<%= ellipsis(title, 30) %> 通常的做法是: 使用 app.locals 来定义: app.loca ...

  4. xcode 编译opencv ios容易出现的错误

    (1)出现 "std::__1::basic_ostream<char, std::__1::char_traits<char> >::flush()"之类 ...

  5. 模态视图(modalTrasitionStyle)如何适应不同的版本

    随着版本的更新,模态视图对版本也有不同的要求,那如何让模态视图适应多版本的要求呢?下面是小编的个人看法! -(void)presentModalVC { modalViewController *mo ...

  6. ACM: 限时训练题解-Epic Professor-水题

    Epic Professor   Dr. Bahosain works as a professor of Computer Science at HU (Hadramout    Universit ...

  7. 【BZOJ】2563: 阿狸和桃子的游戏

    http://www.lydsy.com/JudgeOnline/problem.php?id=2563 题意:给一个n个加权点m条加权边的无向图,两个人轮流拿走一个点,最后使先手得分-后手得分尽量大 ...

  8. C语言(3)

    C语言(3)----数据输入 输入时的关键字为scanf.如我们要从键盘上输入一个数,放在变量a里面,则可以写成scanf("%d",&a); "&&qu ...

  9. DOM的概念及子节点类型【转】

    前言 DOM的作用是将网页转为一个javascript对象,从而可以使用javascript对网页进行各种操作(比如增删内容).浏览器会根据DOM模型,将HTML文档解析成一系列的节点,再由这些节点组 ...

  10. Linux文件锁flock

    Linux文件锁flock 在多个进程同时操作同一份文件的过程中,很容易导致文件中的数据混乱,需要锁操作来保证数据的完整性,这里介绍的针对文件的锁,称之为“文件锁”-flock. flock,建议性锁 ...