python解决循环引用问题

当项目中的模块过多,或功能划分不够清晰时会出现循环引用的问题,如下

有两个模块moduleA 和 moduleB:

#moduleA 
from moduleB import b

def a():
    print 'aaaaaaaa'
    b()

def c():
    print 'cccc'

if __name__ == '__main__':
    a()

#moduleB
from moduleA import c

def b():
    print 'bbbbbbbbbb'
    c()

这种情况下就会出现以下的错误 :

Traceback (most recent call last):
  File "D:/PycharmProjects/untitled2/moduleA.py", line 1, in <module>
    from moduleB import b
  File "D:\PycharmProjects\untitled2\moduleB.py", line 2, in <module>
    from moduleA import c
  File "D:\PycharmProjects\untitled2\moduleA.py", line 1, in <module>
    from moduleB import b
ImportError: cannot import name b

这就是循环引用导致的

导入的实质

导入其实是要将 被导入模块所有的顶格代码都执行一遍,遇到函数和类的定义会作申明。
如果b模块中有这么一句

print 'bbb'

你在a模块impot b时就会 执行 print 'bbb'这一句。

回到循环引用中,首先导入B,进入B中,发现B中又导入了A又回到A中,但是A又导入B这就形成了循环引用。

解法1:直接导入模块名,通过模块调用其中的函数

import moduleB

def a():
    print 'aaaaaaaa'
    moduleB.b()
def c():
    print 'cccccc'


if __name__ == '__main__':
    a()

*************************************
import moduleA

def b():
    print 'bbbbbbbbbb'

    moduleA.c()


具体原因我也不太清楚。

解法2:使用延迟导入(lazy import)

在需要用的函数内部导入,或是在底部导入

moduleB
***************
def b():
    print 'bbbbbbbbbb'

    c()

from moduleA import c

**************************
或者
def b():
    from moduleA import c
    print 'bbbbbbbbbb'
    c()

解法3:重新设计代码结构,将代码和并或者分离

这个要看情况 而定了,将公共功能的代码分离成单独模块,或者将一些具有关系的代码合并成一个模块。

推荐阅读更多精彩内容