目前,我正在研究一个包含子模块的python项目,并使用numpy/scipy。使用Ipython作为交互控制台。不幸的是,我对我现在使用的工作流程不是很满意,我很感激一些建议。

在IPython中,框架是通过一个简单的导入命令加载的。但是,经常需要更改框架的一个子模块中的代码。此时,已经加载了一个模型,我使用IPython与它交互。

现在,框架包含许多相互依赖的模块,也就是说,当框架最初加载时,主模块正在导入和配置子模块。只有使用reload(main_mod.sub_mod)重新加载模块时,才会执行对代码的更改。这很麻烦,因为我需要使用完整的路径单独重新加载所有更改的模块。如果reload(main_module)也会重载所有子模块,但不重载numpy/scipy..


当前回答

IPython自带一些自动重新加载的魔法:

%load_ext autoreload
%autoreload 2

它将在每次执行新行之前重新加载所有更改的模块。这种工作方式与dreload略有不同。一些注意事项适用,输入%autoreload?看看会出什么问题。


如果你想总是启用这个设置,修改你的IPython配置文件~/.ipython/profile_default/ipython_config.py[1]并追加:

c.InteractiveShellApp.extensions = ['autoreload']     
c.InteractiveShellApp.exec_lines = ['%autoreload 2']

通过下面的评论归功于@Kos。

[1] 如果您没有文件~/.ipython/profile_default/ipython_config.py,您需要首先调用ipython配置文件create。或者文件可能位于$IPYTHONDIR。

其他回答

我重新加载的标准做法是在第一次打开IPython后结合这两种方法:

from IPython.lib.deepreload import reload
%load_ext autoreload
%autoreload 2

在此之前加载模块将导致它们不会被重新加载,即使使用手动重新加载(module_name)。我仍然很少遇到类方法不能重新加载的难以解释的问题,我还没有深入研究。

我讨厌为一个长线程添加另一个答案,但我发现了一个解决方案,它支持在%run()上递归重新加载子模块,其他人可能会发现这很有用(反正我有)

删除你想要重载的子模块。iPython中的模块:

In[1]: from sys import modules
In[2]: del modules["mymodule.mysubmodule"] # tab completion can be used like mymodule.<tab>!

现在你的脚本将递归重载这个子模块:

In[3]: %run myscript.py

在你的模块导入之前,包括这些行,其中第一个测试是否已经加载了自动加载扩展:

if 'autoreload' not in get_ipython().extension_manager.loaded:
    %load_ext autoreload
%autoreload 2

import sys
    .
    .
    .

这个怎么样:

import inspect

# needs to be primed with an empty set for loaded
def recursively_reload_all_submodules(module, loaded=None):
    for name in dir(module):
        member = getattr(module, name)
        if inspect.ismodule(member) and member not in loaded:
            recursively_reload_all_submodules(member, loaded)
    loaded.add(module)
    reload(module)

import mymodule
recursively_reload_all_submodules(mymodule, set())

这将有效地重新加载您给它的整个模块和子模块树。你也可以把这个函数放在你的。ipythonrc(我认为)中,这样每次你启动解释器时都会加载它。

注意,上面提到的自动重载只适用于IntelliJ如果你手动保存更改的文件(例如使用ctrl+s或cmd+s)。它好像不能自动保存。