当前位置:  开发笔记 > 编程语言 > 正文

如何卸载(重新加载)Python模块?

如何解决《如何卸载(重新加载)Python模块?》经验,为你挑选了13个好方法。

我有一个长期运行的Python服务器,并希望能够在不重新启动服务器的情况下升级服务.这样做的最佳方法是什么?

if foo.py has changed:
    unimport foo  <-- How do I do this?
    import foo
    myfoo = foo.Foo()

cdleary.. 734

您可以使用reload内置函数在已导入模块时重新加载模块:

from importlib import reload  # Python 3.4+ only.
import foo

while True:
    # Do some things.
    if is_changed(foo):
        foo = reload(foo)

在Python 3中,reload被移动到imp模块.在3.4中,imp被弃用赞成importlib,reload并被添加到后者中.当定位3或更高版本时,要么在调用时引用相应的模块,要么reload导入它.

我认为这就是你想要的.像Django的开发服务器这样的Web服务器使用它,这样您就可以看到代码更改的效果,而无需重新启动服务器进程本身.

引用文档:

重新编译Python模块的代码并重新执行模块级代码,定义一组新的对象,这些对象绑定到模块字典中的名称.扩展模块的init功能不是第二次调用.与Python中的所有其他对象一样,只有在引用计数降为零后才会回收旧对象.模块名称空间中的名称将更新为指向任何新对象或已更改的对象.对旧对象的其他引用(例如模块外部的名称)不会反弹以引用新对象,如果需要,必须在每个命名空间中进行更新.

正如您在问题中提到的,Foo如果Foo类位于foo模块中,您将不得不重建对象.



1> cdleary..:

您可以使用reload内置函数在已导入模块时重新加载模块:

from importlib import reload  # Python 3.4+ only.
import foo

while True:
    # Do some things.
    if is_changed(foo):
        foo = reload(foo)

在Python 3中,reload被移动到imp模块.在3.4中,imp被弃用赞成importlib,reload并被添加到后者中.当定位3或更高版本时,要么在调用时引用相应的模块,要么reload导入它.

我认为这就是你想要的.像Django的开发服务器这样的Web服务器使用它,这样您就可以看到代码更改的效果,而无需重新启动服务器进程本身.

引用文档:

重新编译Python模块的代码并重新执行模块级代码,定义一组新的对象,这些对象绑定到模块字典中的名称.扩展模块的init功能不是第二次调用.与Python中的所有其他对象一样,只有在引用计数降为零后才会回收旧对象.模块名称空间中的名称将更新为指向任何新对象或已更改的对象.对旧对象的其他引用(例如模块外部的名称)不会反弹以引用新对象,如果需要,必须在每个命名空间中进行更新.

正如您在问题中提到的,Foo如果Foo类位于foo模块中,您将不得不重建对象.


不适用于`from m import X`案例.
这个"is_changed"函数来自哪里?我没有看到它的文档,它不在我的Python 3.1.3环境中运行,也不在2.6.4中运行.
@BartoszKP如果`X`不是模块,你可以`导入sys; 重载(sys.modules中[X .__模块__])`
实际上,当您更改文件时,django dev服务器会重新启动...(它重新启动服务器,而不仅仅是重新加载模块)
没有cdleary,Django不能只使用重新加载:http://pyunit.sourceforge.net/notes/reloading.html
对于具有依赖项的模块,重新加载是不够的.请参阅下面的博客:http://stackoverflow.com/a/438845/456878.这曾经让我感到困惑,浪费了10分钟.
仅供参考,重装是邪恶的根源.例如,使用第一个导入创建的对象的所有实例都将引用旧类:`import foo; bar = foo.Bar(); 重新加载(FOO); assert isinstance(bar,foo.Bar)`将引发AssertError
@jedmao @JamesDraper我很确定`is_changed`函数只是一个你必须编写的任意函数; 它不是内置的.例如,它可能会打开与您要导入的模块对应的文件,并使用缓存版本对其进行区分,以查看它是否已更改.
奇怪.答案仅涉及重新加载.卸货怎么样?
如何使用reload进行导入,例如"从abc import someClass as abcClass"... reload(abcClass)抱怨它没有定义.
这个问题的最佳答案如何?它在我的py27和/或py34中不起作用@jedmao你有没有发现"is_changed"的来源?
@JamesDraper nope,从未发现'is_changed`来自哪里......或者我忘记了.这是很久以前的事了.

2> Paul D. Wait..:

在Python 3.0-3.3中,您将使用: imp.reload(module)

该BDFL已经回答了这个问题.

但是,imp在3.4中被弃用,赞成importlib(感谢@Stefan!).

因此,我认为你现在使用了importlib.reload(module),虽然我不确定.


imp.reload(imp)有效吗?
认真的新手很高兴了解Python 2和3之间的关键细微差别.
@LoïcFaure-Lacroix是的,imp可以重装自己.
@LoïcFaure-Lacroix以同样的方式`reload(__ builtins __)`在2.x中有效

3> Gregg Lind..:

如果模块不是纯Python,则删除模块尤其困难.

以下是一些信息:如何真正删除导入的模块?

您可以使用sys.getrefcount()来查找实际的引用数.

>>> import sys, empty, os
>>> sys.getrefcount(sys)
9
>>> sys.getrefcount(os)
6
>>> sys.getrefcount(empty)
3

大于3的数字表示很难摆脱该模块.本土的"空"(不含任何)模块应该在之后进行垃圾收集

>>> del sys.modules["empty"]
>>> del empty

因为第三个引用是getrefcount()函数的工件.


这是正确的解决方案,特别是如果您有一个包含嵌套模块的包.`reload()`只重新加载最顶层的模块,除非你先从sys.modules中删除它,否则不会重新加载其中的任何内容.
我刚刚发现如果模块是包的一部分,你也必须在那里删除它:`setattr(package,"empty",None)`

4> bobince..:

reload(module),但只有它完全是独立的.如果其他任何东西都有对模块(或属于该模块的任何对象)的引用,那么你会得到由旧代码挂起比你预期的更长时间引起的细微和好奇的错误,以及isinstance不能在不同版本的相同的代码.

如果您有单向依赖项,则还必须重新加载依赖于重新加载的模块的所有模块,以去除对旧代码的所有引用.然后递归地重新加载依赖于重新加载的模块的模块.

如果您具有循环依赖关系(例如,在处理重新加载包时非常常见),则必须一次性卸载组中的所有模块.您无法执行此操作,reload()因为它将在刷新依赖项之前重新导入每个模块,从而允许旧引用进入新模块.

在这种情况下,唯一的方法是破解sys.modules,这是一种不受支持的.您必须通过并删除sys.modules下次导入时要重新加载的每个条目,并删除其值为None处理缓存失败的相对导入的实现问题的条目.它并不是非常好,但只要你有一套完全自包含的依赖项,不会将引用留在代码库之外,它就可以使用了.

最好重启服务器.:-)



5> 小智..:
if 'myModule' in sys.modules:  
    del sys.modules["myModule"]


如果您的模块导入它自己的子模块,您可能也需要删除它们.像`[del(sys.modules [mod] for mod in sys.modules.keys()中的东西,如果mod.startswith('myModule.')]`.
+1.我的目标是在python中运行鼻子测试.在我加载了一个模块并重命名了一些函数之后,在调用`nose.run()`时,旧名称仍然存在,即使在`reload(my_module)``%run my_module`之后也是如此.

6> goetzc..:

对于Python 2,使用内置函数reload():

reload(module)

对于Python 2和3.2-3.3,使用从模块imp重新加载:

import imp
imp.reload(module)

imp 不推荐使用,因为3.4版本支持导入库中,所以使用:

import importlib
importlib.reload(module)

要么

from importlib import reload
reload(module)


处理以下任何一种情况:`来自六个import reload_module`(首先需要`pip install six`)

7> Matt Clarkso..:

以下代码允许您兼容Python 2/3:

try:
    reload
except NameError:
    # Python 3
    from imp import reload

您可以reload()在两个版本中使用它,这使事情变得更简单.



8> Joseph Garvi..:

接受的答案不处理来自X导入Y的情况.此代码处理它和标准导入案例:

def importOrReload(module_name, *names):
    import sys

    if module_name in sys.modules:
        reload(sys.modules[module_name])
    else:
        __import__(module_name, fromlist=names)

    for name in names:
        globals()[name] = getattr(sys.modules[module_name], name)

# use instead of: from dfly_parser import parseMessages
importOrReload("dfly_parser", "parseMessages")

在重新加载的情况下,我们将顶级名称重新分配给新重新加载的模块中存储的值,这些值会更新它们.



9> Richie Benda..:

这是重新加载模块的现代方法:

from importlib import reload

只需键入reload(MODULE),替换MODULE为要重新加载的模块的名称.

例如,reload(math)将重新加载数学函数.


或者只是`来自importlib import reload`.然后你可以做`reload(MODULE_NAME)`.不需要此功能.

10> neves..:

如果您不在服务器中,但正在开发并需要经常重新加载模块,这里有一个很好的提示.

首先,确保您使用的是Jupyter Notebook项目中出色的IPython shell.安装Jupyter后,您可以使用ipython,或者jupyter console甚至更好地启动它jupyter qtconsole,这将为您提供一个漂亮的彩色控制台,在任何操作系统中都可以完成代码.

现在在shell中,键入:

%load_ext autoreload
%autoreload 2

现在,每次运行脚本时,都会重新加载模块.

除此之外2,还有autoreload魔术的其他选项:

%autoreload
Reload all modules (except those excluded by %aimport) automatically now.

%autoreload 0
Disable automatic reloading.

%autoreload 1
Reload all modules imported with %aimport every time before executing the Python code typed.

%autoreload 2
Reload all modules (except those excluded by %aimport) every time before
executing the Python code typed.



11> 小智..:

对于那些想要卸载所有模块的人(当在Emacs下的Python解释器中运行时):

   for mod in sys.modules.values():
      reload(mod)

更多信息在重新加载Python模块中.


经过一些修改后,在Python 2.7中运行良好:`if mod and mod .__ name _!="_ _ main__":imp.reload(mod)`
这对我来说效果很好:如果m而不是"__"在m .__ name__而不是imp.is_builtin(m .__ name__),则导入imp [在sys.modules.values()中为m重新加载(m)m

12> 小智..:

Enthought Traits有一个适用于此的模块.https://traits.readthedocs.org/en/4.3.0/_modules/traits/util/refresh.html

它将重新加载已更改的任何模块,并更新正在使用它的其他模块和实例化对象.它在大多数情况下都不能用于__very_private__方法,并且可以阻止类继承,但它在编写PyQt guis时需要重新启动宿主应用程序,或者在Maya或Nuke等程序中运行的东西,这使我节省了大量时间.它可能在20-30%的时间内不起作用,但它仍然非常有用.

Enthought的软件包在它们改变的那一刻不会重新加载文件 - 你必须明确地称它为 - 但如果你真的需要它,那就不应该很难实现



13> PythonMan..:

那些正在使用python 3并从importlib重新加载的人。

如果您遇到问题,例如似乎模块无法重新加载...那是因为它需要一些时间来重新编译pyc(最多60秒)。我编写此提示只是想知道您是否遇到过此类问题。

推荐阅读
mobiledu2402851323
这个屌丝很懒,什么也没留下!
DevBox开发工具箱 | 专业的在线开发工具网站    京公网安备 11010802040832号  |  京ICP备19059560号-6
Copyright © 1998 - 2020 DevBox.CN. All Rights Reserved devBox.cn 开发工具箱 版权所有