什么时候可以导入查找模块?

时间:2016-08-12 07:34:33

标签: python python-2.7

简而言之,这怎么可能发生?

cternus@astarael:~⟫ python
Python 2.7.12 (default, Jun 29 2016, 14:05:02)
[GCC 4.2.1 Compatible Apple LLVM 7.3.0 (clang-703.0.31)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
>>> import backports
>>> import imp
>>> imp.find_module('backports')
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
ImportError: No module named backports

imp模块声称是“用于实现import语句的机制的接口”。如果是这样,为什么import语句会找到backports,但imp.find_module()不能?

对于某些背景:backports声称是“命名空间包”,而不是它本身的包;其他模块(例如backports.shutil_get_terminal_size)位于此命名空间中。这构成了an ultimately-rejected PEP的基础。我问这个问题是因为我有一个this issue的变体,我正试图找出原因。

更奇怪:

>>> backports.__file__
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'module' object has no attribute '__file__'
>>> dir(backports)
['__doc__', '__name__', '__path__']
>>> backports.__path__
['/Library/Python/2.7/site-packages/backports']
>>> import os; os.path.exists(backports.__path__[0])
False

(不,我的系统上没有任何名为backportsbackports.py的文件或目录。)

编辑澄清:我知道这可能代表了我系统的奇怪配置状态。我的问题不是“我怎么能解决这个问题”,而是“怎么可能?”

1 个答案:

答案 0 :(得分:1)

此模块可以由python-configparser APT软件包带来。

  

怎么可能?

这是可能的,因为python-configparser使用路径配置文件.pth文件):

root@ubuntu18.10:/# dpkg -L python-configparser | head | tail -n 1
/usr/lib/python2.7/dist-packages/configparser-3.5.0b2-nspkg.pth

此文件会在解释器启动时由python的 site 模块自动提取,因为它位于/usr/lib/python2.7/dist-packages/中,并具有.pth扩展名。为docs say

  

路径配置文件是名称格式为name.pth的文件,并且存在于上述四个目录之一中...   将执行以import开头的行(后跟空格或制表符)。

文件/usr/lib/python2.7/dist-packages/configparser-3.5.0b2-nspkg.pth包含以下内容:

import sys, types, os;has_mfs = sys.version_info > (3, 5);p = os.path.join(sys._getframe(1).f_locals['sitedir'], *('backports',));importlib = has_mfs and __import__('importlib.util');has_mfs and __import__('importlib.machinery');m = has_mfs and sys.modules.setdefault('backports', importlib.util.module_from_spec(importlib.machinery.PathFinder.find_spec('backports', [os.path.dirname(p)])));m = m or sys.modules.setdefault('backports', types.ModuleType('backports'));mp = (m or []) and m.__dict__.setdefault('__path__',[]);(p not in mp) and mp.append(p)

因此,此代码在python启动时自动执行。美化了一下,看起来像这样:

import sys, types, os

has_mfs = sys.version_info > (3, 5)

p = os.path.join(sys._getframe(1).f_locals['sitedir'], *('backports',))
importlib = has_mfs and __import__('importlib.util')
has_mfs and __import__('importlib.machinery')

m = has_mfs and sys.modules.setdefault('backports', importlib.util.module_from_spec(importlib.machinery.PathFinder.find_spec('backports', [os.path.dirname(p)])))

m = m or sys.modules.setdefault('backports', types.ModuleType('backports'))

mp = (m or []) and m.__dict__.setdefault('__path__',[])

(p not in mp) and mp.append(p)

它的作用(至少在python 2上如此)是:它手动通过调用types.ModuleType构造函数来创建模块对象(这就是为什么它看起来像<module 'backports' (built-in)>)并放入并以sys.modulessys.modules.setdefault('backports', types.ModuleType('backports'))。将其添加到sys.modules后,import backports只会返回该对象。

__path__提示

root@ubuntu18.10:/# python -c 'import backports; print backports.__path__'
['/usr/lib/python2.7/dist-packages/backports']
root@ubuntu18.10:/# dpkg -S /usr/lib/python2.7/dist-packages/backports
python-configparser: /usr/lib/python2.7/dist-packages/backports
  

我没有名为backports的文件或目录

在Ubuntu上,该软件包带来了 /usr/lib/python2.7/dist-packages/backports,如上所示,所以我不确定为什么没有它。也许这是另一个与MacOS行为类似/不同的软件包/您在调试问题时删除了该目录吗?