1.什么时循环导入
其实就是我们有一个A.py文件和一个B.py文件,我们在A.py文件导入B模块的某个名称,在B.py
文件中又导入了A模块的某个名称,但是由于我们把from···import都卸载了这两个文件的开头,所以就会出现循环导入的问题。
我们先执行哪一个.py文件,如果出现了循环导入的问题,程序就会报错这两个模块里某一个模块没有另一个模块要找的名称
# m1.py
print('from m1.py')
from m2 import x
y = 'm1'
- 创建m2的名称空间
- 执行m2.py,将执行产生的名字丢到m2.py
- 在当前执行文件中拿到m2.x
# m2.py
print('from m2.py')
from m1 import y
x = 'm2'
- 创建m1的名称空间
- 执行m1.py,将执行产生的名字丢到m1.py
- 在当前执行文件中拿到m1.y
# run.py
import m1
- 创建m1的名称空间
- 执行m1.py,将执行产生的名字丢到m1.py
- 在当前执行文件中拿到m1
如果运行run.py,则会报错ImportError: cannot import name 'y'
如果运行m1.py,则会报错ImportError: cannot import name 'x'
如果运行m2.py,则会报错ImportError: cannot import name 'y'
2.解决方案
我们可以使用函数定义阶段只识别语法的特性解决循环导入的问题,我们也可以从本质上解决循环导入的问题,但是最好的解决方法是不要出现循环导入。
方案一
使用函数定义阶段只识别语法的特性,在这两个模块都定义函数,并且需要把from····import写在最开始的位置
# m1.py
print('from m1.py')
def func1():
from m2 import x
print(x)
y = 'm1'
# m2.py
print('from m2.py')
def func1():
from m1 import y
print(y)
x = 'm2'
方案二
就是把我们两个模块需要从对方模块中找到的名字提前和from····import调整一下语句的先后顺序
# m1.py
print('from m1.py')
y = 'm1'
from m2 import x
# m2.py
print('from m2.py')
x = 'm2'
from m1 import y