17

我正在尝试使用 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")
4

1 回答 1

29

尝试这个:

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请参阅forimportlib的以下评论:

# 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并获得AttributeError, util(连同其他子模块)被加载/初始化后,现在您可以访问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']
于 2016-09-23T12:44:59.363 回答