我有一些代码分布在多个文件中,试图从彼此导入,如下所示:

main.py:

from entity import Ent

entity.py:

from physics import Physics
class Ent:
    ...

physics.py:

from entity import Ent
class Physics:
    ...

然后从main.py运行,得到以下错误:

Traceback (most recent call last):
File "main.py", line 2, in <module>
    from entity import Ent
File ".../entity.py", line 5, in <module>
    from physics import Physics
File ".../physics.py", line 2, in <module>
    from entity import Ent
ImportError: cannot import name Ent

我假设错误是由于导入实体两次-一次在main.py和物理。py -但我如何解决这个问题?


参见在Python中使用相互导入或循环(循环)导入时会发生什么?了解WRT循环导入所允许的内容以及导致问题的原因。参见为什么循环导入看起来在调用堆栈中更上一层,但随后在更下一层引发ImportError ?有关问题发生的原因和方式的技术细节。


当前回答

在我的例子中,我在一个Jupyter笔记本上工作,这是因为当我在工作文件中定义类/函数时,导入已经被缓存了。

我重新启动了Jupyter内核,错误消失了。

其他回答

也不是直接相关的OP,但重启PyCharm Python控制台失败后,添加一个新对象到一个模块,也是一个很好的方式,得到一个非常混乱的ImportError:不能导入名称…

令人困惑的是,PyCharm会在控制台中自动完成导入,但导入随后会失败。

问题很明显:实体和物理模块中名称之间的循环依赖关系。

无论导入整个模块还是仅仅导入一个类,都必须加载名称。

看这个例子:

# a.py
import b
def foo():
  pass
b.bar()
# b.py
import a
def bar():
  pass
a.foo()

这将被汇编成:

# a.py
# import b
# b.py
# import a # ignored, already importing
def bar():
  pass
a.foo()
# name a.foo is not defined!!!
# import b done!
def foo():
  pass
b.bar()
# done!

只要做一点小小的改变,我们就能解决这个问题:

# a.py
def foo():
  pass
import b
b.bar()
# b.py
def bar():
  pass
import a
a.foo()

这将被汇编成:

# a.py
def foo():
  pass
# import b
# b.py
def bar():
  pass
# import a # ignored, already importing
a.foo()
# import b done!
b.bar()
# done!

逻辑清楚是很重要的。出现这个问题,是因为引用变成了死循环。

如果你不想改变逻辑,你可以把导致ImportError的import语句放到文件的另一个位置,例如末尾。

a.py

from test.b import b2

def a1():
    print('a1')
    b2()

b.py

from test.a import a1

def b1():
    print('b1')
    a1()

def b2():
    print('b2')

if __name__ == '__main__':
    b1()

导入错误:ImportError:不能导入名称'a1'

但是如果我们改变test的位置。b import b2 in A,如下所示:

a.py

def a1():
    print('a1')
    b2()

from test.b import b2

我们可以得到我们想要的:

b1
a1
b2

这是一个循环依赖。 我们可以在需要的地方使用import模块或类或函数来解决这个问题。 如果我们使用这种方法,就可以修复循环依赖关系

A.py

from B import b2
def a1():
    print('a1')
    b2()

B.py

def b1():
   from A import a1
   print('b1')
   a1()

def b2():
   print('b2')
if __name__ == '__main__':
   b1() 

试试这个解决方案:重命名你的工作python脚本

你不应该用你导入的其他模块的名字来命名你当前的python脚本,因为你会得到这个错误。

例子:

您正在medicaltorch.py中工作 在这个脚本中,你有:from medicaltorch import X,其中medicaltorch应该是一个单独安装的模块

这将失败的ImportError,因为2件事涉及医疗火炬

因此,只需在1中重命名您的工作python脚本。