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

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 ?有关问题发生的原因和方式的技术细节。


当前回答

在我的例子中,只是错过了文件名:

from A.B.C import func_a (x)

from A.B.C.D import func_a (O)

其中D为file。

其他回答

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

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

如果你从file1.py导入file1.py并使用这个:

if __name__ == '__main__':
    # etc

file1.py中低于该值的变量不能导入到file1.py中,因为__name__不等于__main__!

如果你想从file1.py导入一些东西到file2.py,你需要在file1.py中使用这个:

if __name__ == 'file1':
    # etc

如果有疑问,使用assert语句确定__name__=='__main__'

你有循环依赖的导入。在定义类Ent之前从实体导入physics.py,并且physics尝试导入已经初始化的实体。从实体模块中移除对物理的依赖。

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

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

看这个例子:

# 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!

跟踪导入错误的一种方法是一步一步地尝试在每个导入的文件上运行python来跟踪坏的文件。

你会得到这样的结果: python。/ main.py ImportError:不能导入名称A 然后你启动: python。/模块/ a.py ImportError:不能导入名称B 然后你启动: python。/模块/ b.py ImportError:不能导入名称C(一些不存在的模块或其他错误)