【python学习笔记】如何说明import语句对模块导入的检查?

问题来自 learning python 中的一个练习题

1,为什么交互模式导入 recur1 会报错?
2,为什么交互模式导入 recur2 不会报错?
3,作为文件运行 recur1.py 不会报错?
4,作为文件运行 recur2.py 会报错?


recur1 和 recur2 文件内容如下

# recur1.py
X = 1
import recur2
Y = 2

# recur2.py
from recur1 import X
from recur1 import Y
解答:

在交互模式中:import recur1 # 报错找不到 Y

因为先导入recur1时,python先为模块recur1分配内存空间(假设就像C的内存分配一样),地址存到sys.modules的字典中,然后执行 X = 1,然后执行 import recur2, 同样会存模块名recur2到 sys.modules 字典中,然后执行 from recur1 import X,因为 recur1 在字典中,且 X 已经执行过赋值,所以没问题(假设 X = 1 不会重复执行,因为模块导入也不会重复执行代码),再执行 import recur2,因为 recur2 已在字典中,但是没有执行完(因为 from 会要求完整执行模块,以便拷贝出变量?),所以接着执行 from recur1 import Y,因为 Y = 2,还没有执行,所以报错 Y 不存在。

在交互模式中:当 import recur2 时

recur2 模块先存到字典中,执行 from recur1 import X,存 recur1 模块到字典中,执行 X = 1,执行 import recur2,因为recur2已在字典中,正常(为什么说正常下文有说明)执行无报错,执行 Y = 2,完了再回到from recur1 import X,拷贝出 X,执行 from recur1 import Y,Y 已经被赋值了,所以不会报错。


为什么说正常?

如何证明 import recur2,只是检查 sys.modules 字典存在了 recur2 模块,而不管 recur2 模块是否完整?

将 recur2.py 文件清空后改成如下内容:

import sys
print('in recur2.py:', sys.modules['recur2'])

print('import again:')
import recur2

print('make a mistake:')
print(a)
print('end')

在交互模式中运行:

>>> import recur2
(1)in recur2.py: 'recur2' from 'D:\\python_code\\recur2.py'>  
(2)import again:
(3)make a mistake:
(4)Traceback (most recent call last):
  File "", line 1, in 
  File "D:\python_code\recur2.py", line 8, in 
    print(a)
NameError: name 'a' is not defined
>>> recur2
Traceback (most recent call last):
  File "", line 1, in 
NameError: name 'recur2' is not defined
>>>
结论:

(1) 说明在字典中存储了 recur2 模块名。
(2) 说明再次导入时(实际导入 recur2 过程并没有完成)没有报错。
直到 (4) 报错,说明 import 导入的确只是检查字典中有无 recur2 模块。
当再一次运行 >>> recur2 时,并没有成功创建 recur2 模块,说明 NameError 异常影响了导入行为。

说明:以上心得若有不足、错误之处欢迎各位指正。

你可能感兴趣的:(Python,python)