python模块引用常出现的错误_python模块交叉引用的错误案例

网络上有一则关于python模块交叉引用的错误案例说明,很经典。先给出链接:https://wiki.woodpecker.org.cn/moin/MiscItems/2008-11-25。

我还是需要先对这个问题做一个说明,有如下两个python文件,代码如下:

$ cat a.py

from b import d

class c: pass

$ cat b.py

from a import c

class d: pass

执行a.py或b.py,会出现如下错误:

$ python3 a.py

Traceback (most recent call last):

File "a.py", line 1, in from b import d

File "/home/xinlin/test/b.py", line 1, in

from a import c

File "/home/xinlin/test/a.py", line 1, in

from b import d

ImportError: cannot import name 'd' from 'b' (/home/xinlin/test/b.py)

$ python3 b.py

Traceback (most recent call last):

File "b.py", line 1, in from a import c

File "/home/xinlin/test/a.py", line 1, in

from b import d

File "/home/xinlin/test/b.py", line 1, in

from a import c

ImportError: cannot import name 'c' from 'a' (/home/xinlin/test/a.py)

出现ImportError错误!这就是模块交叉引用造成的错误。如何这个错误,就要深入python的from ... import ...机制。

这段解释很详细,括号中的为我自己加的内容:

1、执行A.py中的from B import D

由于是执行的python A.py,所以在sys.modules中并没有存在,

首先为B.py创建一个module对象(),

注意,这时创建的这个module对象是空的,里边啥也没有,(sys.modules是一个dict对象,module B为空,表示这个dict的module B这个key也不存在)

在Python内部创建了这个module对象之后,就会解析执行B.py,其目的是填充这个dict。

2、执行B.py中的from A import C

在执行B.py的过程中,会碰到这一句,

首先检查sys.modules这个module缓存中是否已经存在了,

由于这时缓存还没有缓存,

所以类似的,Python内部会为A.py创建一个module对象(),

然后,同样地,执行A.py中的语句

3、再次执行A.py中的from B import D

这时,由于在第1步时,创建的对象已经缓存在了sys.modules中,

所以直接就得到了,

但是,注意,从整个过程来看,我们知道,这时还是一个空的对象,里面啥也没有,

所以从这个module中获得符号"D"的操作就会抛出异常。

如果这里只是import B,由于"B"这个符号在sys.modules中已经存在,所以是不会抛出异常的。

修正错误的方法,就是不要使用from ... import ...这个语句,直接import即可!

我始终觉着这是个设计问题,python a.py的时候,就不应该在其它模块中 import a!交叉引用本来就是个设计问题,层次不明。

-- EOF --

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值