关于SO上的重新导入有很多问题和答案,但如果不了解其背后的机制,这一切似乎都非常直观。
如果您导入模块,更改内容,然后尝试再次导入,您会发现第二次导入无效:
>>> import foo # foo.py contains: bar = 'original'
>>> print foo.bar
original
>>> # edit foo.py and change to: bar = 'changed'
>>> import foo
>>> print foo.bar
original
当我发现reload
时,我是一个非常开心的露营者:
>>> reload(foo)
>>> print foo.bar
changed
但是,当您从模块导入项目而不导入模块本身时,没有简单的解决方案:
>>> from foo import baz
>>> print baz
original
>>> # change foo.py from baz = 'original' to baz = 'changed'
>>> from foo import baz
>>> print baz
original
>>> reload(foo)
Traceback (most recent call last):
File "<pyshell#10>", line 1, in <module>
reload(foo)
NameError: name 'foo' is not defined
为什么Python不会在为其提供新的import
语句时更新导入的项目?
答案 0 :(得分:10)
导入模块时,它会缓存在sys.modules
中。在同一会话中再次导入同一模块的任何尝试都只返回其中包含的现有模块。当从多个位置导入模块时,这可以加快整体体验。它还允许模块在所有导入之间共享自己的对象,因为每次都返回相同的模块。
如前所述,您可以使用reload
重新导入整个模块。检查文档中的警告,因为即使这不是万无一失的。
从模块导入特定项目时,将按上述方式导入整个模块,然后将请求的对象放在命名空间中。 reload
不起作用,因为这些对象不是模块,并且您从未收到对模块本身的引用。解决方法是获取对模块的引用,重新加载它,然后重新导入:
>>> from foo import baz
>>> print baz
original
>>> # change foo.py from baz = 'original' to baz = 'changed'
>>> import foo
>>> reload(foo)
>>> from foo import baz
>>> print baz
changed
答案 1 :(得分:2)
“为什么”的简单解释是import语句涉及两个操作:加载模块(即,在其中运行代码),以及将名称导入导入模块的命名空间。重新导入仅重做第二个操作。原因是第一次操作在计算和内存资源方面可能很昂贵。
提供 reload
作为重做第一个操作的方法,但是,如您所见,它需要模块名称。但是,不要误以为from foo import bar
没有加载整个模块。确实如此。所有模块的代码都会运行,只是你只能引用你明确导入的名称(例如,bar
)。因此,导入模块与不导入模块之间几乎没有什么区别;唯一的区别是命名空间问题。因此,如果您认为可能需要重新加载模块foo
,则应继续执行import foo
。如果需要,可以执行此操作以刷新导入的名称:
import foo
from foo import bar
# later...
reload(foo)
from foo import bar
还有另一个原因是你不能重新导入个人名字,这就是Python无法改变所引用的对象;它只能重新绑定名称。假设你这样做:
from foo import bar
newBar = bar
from foo import bar
即使第二个导入确实刷新了bar的值,它也永远不会刷新newBar的值,因为foo和bar都没有任何方法知道你创建了一个额外的名称来引用bar。避免重新绑定个别名称可以防止您陷入此陷阱;如果人们认为重新导入会刷新名称,那么很容易忘记它仍然不会刷新对象。