26

我想将不同类中的代码分开并将它们放到不同的文件中。然而,这些类是相互依赖的。

主要.py:

from lib import A, B

def main():
    a = A()
    b = B()
    a.hello()
    b.hello()

if __name__ == '__main__':
    main()

lib/_初始化_.py

from a import A
from b import B

库/a.py:

import lib.B

class A():
    def __init__(self):
        print "A"

    def hello(self):
        print "hello A"
        b = B()

库/b.py:

import lib.A

class B():
    def __init__(self):
        print "B"

    def hello(self):
        print "hello B"
        a = A()

有可能在 Python 中做到这一点吗?

编辑:

我收到此错误消息:

pydev debugger: starting
Traceback (most recent call last):
  File "eclipse-python/plugins/org.python.pydev_2.7.1.2012100913/pysrc/pydevd.py", line 1397, in <module>
    debugger.run(setup['file'], None, None)
  File "eclipse-python/plugins/org.python.pydev_2.7.1.2012100913/pysrc/pydevd.py", line 1090, in run
    pydev_imports.execfile(file, globals, locals) #execute the script
  File "main.py", line 2, in <module>
    from lib import A, B
  File "lib/__init__.py", line 1, in <module>
    from a import A
  File "lib/a.py", line 1, in <module>
    import lib.B
ImportError: No module named B
4

4 回答 4

30

您可以在 hello 函数中导入其他模块,而不是导入顶部的模块。

class B():
    def __init__(self):
        print "B"

    def hello(self):
        from lib import A
        print "hello B"
        a = A()
于 2013-06-21T00:40:42.023 回答
10

当你有两个相互依赖的类时,通常意味着它们要么真的属于同一个模块,要么意味着你的耦合太紧密,应该使用依赖注入来解决。

现在确实有几个极端情况,从函数内部导入是“最差”的解决方案,但这仍然是您应该尽可能避免的事情。

于 2013-06-21T06:08:29.313 回答
9

您的主要问题是您正在尝试导入一个类,但使用仅适用于导入模块的语法。具体来说,如果是在模块中定义的类(并导入到 的顶级命名空间中),import lib.A则永远不会工作。Alib.alib

我建议你避免使用from _ import _语法,除非你真的需要它。这使得依赖关系更容易解决:

lib/a.py

import lib.b # note, we're not importing the class B, just the module b!

class A():
    def foo(self):
        return lib.b.B() # use the class later, with a qualified name

lib/b.py

import lib.a # again, just import the module, not the class

class B():
    def foo(self):
        return lib.a.A() # use another qualified name reference

lib/__init__.py

from a import A # these imports are fine, since the sub-modules don't rely on them
from b import B # they can be the public API for the A and B classes

如果您不希望a并且b依赖于其 package 的名称,您也可以使用相对模块导入lib

这肯定会起作用,因为 A 或 B 类实际上都不需要另一个存在才能被定义。只有在它们被导入之后,A 的实例才需要了解 B 类(反之亦然)。

如果其中一个类从另一个类继承,或者在顶层使用了另一个类的实例,则您需要更加小心首先加载哪个模块,否则它可能仍然会中断。

于 2013-06-21T07:24:55.293 回答
2

如果你只想导入一次,你可以在类的构造函数中导入它并使变量成为全局变量:

class B():
    def __init__(self):
        global A
        from lib import A
        print "B"

    def hello(self):
        print "hello B"
        a = A()

这会将 A 导入到全局变量中,并使其在模块内以可访问的形式出现。

于 2013-06-21T01:22:46.663 回答