在python中所有加载到内存中的模块都存在于sys.modules
当 import 一个模块时首先会在这个列表中查找是否已经加载了此模块:
如果加载了则只是将模块的名字加入到正在调用 import 的模块的 Local 名字空间中。
如果没有加载则从 sys.path 目录中按照模块名称查找模块文件,模块可以是py、pyc、pyd,找到后将模块载入内存,并加到 sys.modules 中,
并将名称导入到当前的 Local名字空间
一个模块不会重复载入。多个不同的模块都可以用 import 引入同一个模块到自己的 Local 名字空间,其实背后的 PyModuleObject 对象只有一个。
注意
例如:本模块导入 A 模块(import A),A 中又 import B,B 模块又可以 import 其他模块……
注意:各个模块的 Local 名字空间是独立的;
对于上面的例子,本模块 import A 之后本模块只能访问模块 A,不能访问模块 B 及其他模块。虽然模块 B 已经加载到内存了,如果访问还要再明确的在本模块中 import B。
例如:文件[ A.py ]
from B import D
class C:pass
文件[ B.py ]
from A import C
class D:pass
为什么执行 A 的时候不能加载 D 呢?
如果将 A.py 改为:import B 就可以了。
这是怎么回事呢?
RobertChen:这跟Python内部 import 的机制是有关的,具体到 from B import D,Python 内部会分成几个步骤:
(1)在 sys.modules 中查找符号 “B”
(2)如果符号 B 存在,则获得符号 B 对应的 module 对象。
从
(3)如果符号 B 不存在,则创建一个新的 module 对象
执行 B.py 中的表达式,填充
从
所以这个例子的执行顺序如下:
1、执行 A.py 中的 from B import D 由于是执行的 python A.py,所以在 sys.modules 中并没有
2、执行 B.py中的from A import C 在执行B.py的过程中,会碰到这一句,首先检查sys.modules这个module缓存中是否已经存在
3、再次执行A.py中的from B import D 这时,由于在第1步时,创建的
只要一个文件夹下面有个 __init__.py 文件,那么这个文件夹就可以看做是一个包。
包导入的过程和模块的基本一致,只是导入包的时候会执行此包目录下的 __init__.py 而不是模块里面的语句了。
如果只是单纯的导入包,而包的 __init__.py 中又没有明确的其他初始化操作,那么此包下面的模块是不会自动导入的