from module import symbol检查模块是否在sys.modules中,如果是,则返回它。 如果要导入以从磁盘重新加载模块,则可以先删除__dict__中的相应密钥。
有一个from module import symbol内置函数,给定模块对象,该函数将从磁盘重新加载它,并将其放置在sys.modules中。编辑-实际上,它将从磁盘上的文件重新编译代码,然后重新评估它。 现有模块的__dict__。可能与创建新模块对象有很大不同。
迈克·格雷厄姆(Mike Graham)是对的。 如果甚至有几个活动对象引用了您不再想要的模块内容,那么要正确地重新加载就很难了。 现有对象仍将引用从其实例化的类,这是一个明显的问题,但是通过from module import symbol创建的所有引用仍将指向模块旧版本中的任何对象。 许多细微错误的事情都是可能的。
编辑:我同意重新启动解释器是迄今为止最可靠的事情。 但是出于调试目的,我想您可以尝试以下操作。 我确定在某些极端情况下这是行不通的,但是如果您对包中的模块加载没有做任何疯狂的事情(否则),这可能会很有用。
def reload_package(root_module):
package_name = root_module.__name__
# get a reference to each loaded module
loaded_package_modules = dict([
(key, value) for key, value in sys.modules.items()
if key.startswith(package_name) and isinstance(value, types.ModuleType)])
# delete references to these loaded modules from sys.modules
for key in loaded_package_modules:
del sys.modules[key]
# load each of the modules again;
# make old modules share state with new modules
for key in loaded_package_modules:
print 'loading %s' % key
newmodule = __import__(key)
oldmodule = loaded_package_modules[key]
oldmodule.__dict__.clear()
oldmodule.__dict__.update(newmodule.__dict__)
我对此进行了简短的测试:
import email, email.mime, email.mime.application
reload_package(email)
印刷:
reloading email.iterators
reloading email.mime
reloading email.quoprimime
reloading email.encoders
reloading email.errors
reloading email
reloading email.charset
reloading email.mime.application
reloading email._parseaddr
reloading email.utils
reloading email.mime.base
reloading email.message
reloading email.mime.nonmultipart
reloading email.base64mime