使用importlib.util检查库时出错

时间:2016-09-23 12:36:22

标签: python python-3.x python-importlib

我正在尝试使用importlib库来验证在Python 3.5.2中执行脚本的计算机上是否安装了nmap库

我正在尝试使用importlib.util.find_spec("nmap")但收到以下错误。

>>> import importlib
>>> importlib.util.find_spec("nmap")
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: module 'importlib' has no attribute 'util'

有人能告诉我哪里出错了吗?

修改

我能够使用以下代码使函数正常工作。

#!/usr/bin/pythonw

import importlib
from importlib import util

#check to see if nmap module is installed
find_nmap = util.find_spec("nmap")
if find_nmap is None:
    print("Error")

1 个答案:

答案 0 :(得分:12)

试试这个:

from importlib import util
util.find_spec("nmap")

我打算调查,但说实话,我不知道为什么一个有效,另一个没有。另外,请观察以下交互式会话:

>>> import importlib
>>> importlib.util
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: module 'importlib' has no attribute 'util'
>>> from importlib import util
>>> util
<module 'importlib.util' from '/usr/lib/python3.5/importlib/util.py'>
>>> importlib.util
<module 'importlib.util' from '/usr/lib/python3.5/importlib/util.py'>

所以...是啊。我相信这对某人来说是完全合理的,但不是对我而言。一旦我搞清楚,我会更新。

更新

将此比较为:

>>> import datetime
>>> datetime
<module 'datetime' from '/usr/lib/python3.5/datetime.py'>
>>> datetime.datetime
<class 'datetime.datetime'>

我认为不同的是,在这种情况下,第一个datetime是一个模块,第二个是一个类,而在importlib.util情况下,两个都是模块。所以,除非加载了两个模块的代码,否则module.module可能不正常,而module.class是正常的,因为在导入模块时会加载类代码。

更新#2

不,似乎在很多情况下module.module都没问题。例如:

>>> import urllib
>>> urllib
<module 'urllib' from '/usr/lib/python3.5/urllib/__init__.py'>
>>> urllib.error
<module 'urllib.error' from '/usr/lib/python3.5/urllib/error.py'>

所以也许这是importlib特有的东西。

更新#3

正如 @kfb 在评论中指出的那样,它似乎与importlib具体相关。请参阅__init__.py for importlib

中的以下评论
# Until bootstrapping is complete, DO NOT import any modules that attempt
# to import importlib._bootstrap (directly or indirectly). Since this
# partially initialised package would be present in sys.modules, those
# modules would get an uninitialised copy of the source version, instead
# of a fully initialised version (either the frozen one or the one
# initialised below if the frozen one is not available).

importlib/util.py 导入importlib._bootstrap所以我认为这是真的。如果我的理解是正确的,当您执行import importlib时,子模块将被初始化,但不会为您导入的importlib模块对象进行初始化。此时,如果您执行dir(importlib),则不会看到util。有趣的是,之后,您尝试访问importlib.util并获得AttributeErrorutil(以及其他子模块)被加载/初始化,现在您< em>可以访问importlib.util

>>> import importlib
>>> dir(importlib)
['_RELOADING', '__all__', '__builtins__', '__cached__', '__doc__', '__file__', '__import__', '__loader__', '__name__', '__package__', '__path__', '__spec__', '_bootstrap', '_bootstrap_external', '_imp', '_r_long', '_w_long', 'find_loader', 'import_module', 'invalidate_caches', 'reload', 'sys', 'types', 'warnings']
>>> importlib.util
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: module 'importlib' has no attribute 'util'
>>> importlib.util
<module 'importlib.util' from '/usr/lib/python3.5/importlib/util.py'>
>>> dir(importlib)
['_RELOADING', '__all__', '__builtins__', '__cached__', '__doc__', '__file__', '__import__', '__loader__', '__name__', '__package__', '__path__', '__spec__', '_bootstrap', '_bootstrap_external', '_imp', '_r_long', '_w_long', 'abc', 'find_loader', 'import_module', 'invalidate_caches', 'machinery', 'reload', 'sys', 'types', 'util', 'warnings']