通过实验,我验证了当同一目录中同时存在编译的extension.pyd(或.so)和Plain时extension.py,.pyd首先导入文件;的.py,如果只输入.pyd文件未找到:
extension.pyd
.so
extension.py
.pyd
.py
In [1]: import extension In [2]: extension.__file__ Out[2]: 'extension.pyd' In [3]: import glob; glob.glob("extension.py*") Out[3]: ['extension.py', 'extension.pyd']
是否保证所有版本的Python都相同,并且我可以依靠它向.py仅在.pyd未找到文件时执行的文件添加逻辑吗?
FWIW,我找不到参考说明,必须在py文件之前加载扩展,因此将其作为实现细节可能更安全(除非有人提供参考)。即使此细节对于所有版本都稳定到至少2.7。
导入模块时,它将首先在缓存中查找(即sys.modules),如果尚未在其中,sys.meta_path则使用from中的查找器。通常,sys.meta_path由BuiltinImporter,FrozenImporter和组成PathFinder,其中PathFinder负责在磁盘/ python-path上查找模块。
sys.modules
sys.meta_path
BuiltinImporter
FrozenImporter
PathFinder
PathFinder提供了一些缓存功能,以加速查找,但它基本上代表所述搜索到钩从sys.path_hooks-概述可以在例如发现PEP 302。
sys.path_hooks
通常情况下,sys.path_hooks包括zipimporter,这使压缩文件的导入可能,和一个包裹 FileFinder,这是整个进口机械的工作马。
zipimporter
FileFinder
FileFinder尝试不同的出就足够了(即.so,.py,.pyc)在给定的顺序,这是通过建立_get_supported_file_loaders()-method:
.pyc
_get_supported_file_loaders()
def _get_supported_file_loaders(): """Returns a list of file-based module loaders. Each item is a tuple (loader, suffixes). """ extensions = ExtensionFileLoader, _imp.extension_suffixes() source = SourceFileLoader, SOURCE_SUFFIXES bytecode = SourcelessFileLoader, BYTECODE_SUFFIXES return [extensions, source, bytecode]
可以看到:
显然,sys.meta_path以及sys.path_hooks可以以某种方式进行操纵,这将建立负载优先级的任意顺序。
作为个人说明:我将尽量避免py和so / pyd文件彼此相邻的情况。