简而言之,这怎么可能发生?
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
(不,我的系统上没有任何名为backports
或backports.py
的文件或目录。)
编辑澄清:我知道这可能代表了我系统的奇怪配置状态。我的问题不是“我怎么能解决这个问题”,而是“怎么可能?”
答案 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.modules
到sys.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行为类似/不同的软件包/您在调试问题时删除了该目录吗?