导入不存在的包

kjs*_*roo 5 python python-import python-2.7

我以前从未见过像这样的导入问题.我删除了一个目录,site-packages相应的包仍然是可导入的.

python2
> import google
> print(google.__path__)
['/home/bamboo/.local/lib/python2.7/site-packages/google']
Run Code Online (Sandbox Code Playgroud)

但是这个目录实际上并不存在

ls: cannot access /home/bamboo/.local/lib/python2.7/site-packages/google: No such file or directory
Run Code Online (Sandbox Code Playgroud)

我已经删除了我所知道的与之相关的所有内容,但仍然必须有一些东西悬而未决.

挖掘更深层次,我试图重新加载google.

python2
> import google;
> reload(google);
ImportError: No module named google
Run Code Online (Sandbox Code Playgroud)

所以显然它认识到重新加载了.

退房,sys.modules你得到

python2
> import sys
> print(sys.modules)
{'google': <module 'google' (built-in)>, 'copy_reg': <module 'copy_reg' from '/usr/lib/python2.7/copy_reg.pyc'> ...
Run Code Online (Sandbox Code Playgroud)

这表明它显然google是内置的.

关于动机的注意事项:通常这类问题会很奇怪,但不是一个显示限制因素.对我来说问题是该google软件包正在屏蔽同名的不同软件包.

sna*_*erb 4

tl,dr:用于pip完全卸载Google软件包。

这里有两个问题:

  • 谷歌包的奇怪导入/重新加载行为
  • 删除谷歌包

导入/重新加载行为

我可以通过安装 (Google) protobuf包来重现导入/重新加载行为(许多 Google 包的行为方式相同)。

$ mktmpenv -p $(which python2)
...
$ python --version
Python 2.7.13
$ pip install protobuf
...
Installing collected packages: six, protobuf
Successfully installed protobuf-3.5.1 six-1.11.0

>>> import google
>>> print google.__path__
['~/virtual-envs/tmp-66cd9b4d01a8dec6/lib/python2.7/site-packages/google']
>>> import sys
>>> print sys.modules['google']
<module 'google' (built-in)>
>>> reload(google)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
ImportError: No module named google
Run Code Online (Sandbox Code Playgroud)

我怀疑这里发生的事情是 Google 更喜欢将所有 Google 软件包安装在单个google软件包下,但该软件包并非设计为可导入的,因此会出现意外的重新加载行为。但是按名称导入子包可以按预期工作:

>>> import protobuf
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
ImportError: No module named protobuf

>>> from google import protobuf
>>> protobuf.__path__
['~/virtual-envs/tmp-66cd9b4d01a8dec6/lib/python2.7/site-packages/google/protobuf']
>>> reload(protobuf)
<module 'google.protobuf' from '~/virtual-envs/tmp-66cd9b4d01a8dec6/lib/python2.7/site-packages/google/protobuf/__init__.pyc'>
>>> 
Run Code Online (Sandbox Code Playgroud)

删除谷歌包

问题指出:

我从 site-packages 中删除了一个目录,相应的包仍然可以导入。

这也可以重现:

($ rm -rf ~/virtual-envs/tmp-66cd9b4d01a8dec6/lib/python2.7/site-packages/google
$  python
>>> import google
>>> print google.__path__
['~/virtual-envs/tmp-66cd9b4d01a8dec6/lib/python2.7/site-packages/google']
>>> 
Run Code Online (Sandbox Code Playgroud)

这里的问题是,仅仅删除google目录及其内容不足以完全卸载现有的任何 Google 软件包。

site-packages 目录仍然包含文件protobuf-3.5.1-py2.7-nspkg.pth,其中包含以下代码(为了可读性分成单独的行,原始是单行分号分隔的语句):

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

线路

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

如果模块尚不存在,则正在创建该google模块- 这就是为什么即使在删除目录后该模块仍然可以导入的原因。sys.modulesgoogle

删除模块的正确方法google是使用以下命令卸载 google 软件包pip

pip uninstall protobuf

如果pip在构建环境中不可用,则需要识别站点包中的任何相关文件和文件夹( *dist-info/、 )并手动删除它们。*.pth