基本概念
Python中的包,即包含 __init__.py 文件的文件夹。
对于Python的包内导入,即包内模块导入包内模块,存在绝对导入和相对导入问题。
普通 Python 模块的搜索路径
1. 在当前模块所在路径中搜索导入模块
2. 在环境变量PYTHONPATH指定的路径列表中搜索导入模块
3. 在sys.path指定的路径列表中搜索导入模块
Python import 的步骤
Python 所有加载的模块信息都存放在 sys.modules字典结构中,当import一个模块时,会按如下步骤来进行
1. 如果importA,检查sys.modules中是否已经有A,如果有则不加载,如果没有则为A创建module对象,并加载A,即可以重复导入,但只加载一次。
2. 如果from A import B,先为A创建module对象,再解析 A,从中寻找 B 并填充到 A 的 __dict__ 中。
相对导入与绝对导入
绝对导入的格式为 import A.B 或 from A import B,相对导入格式为 from .A import B 或 from ..X import Y,. 代表当前模块,.. 代表上层模块,… 代表上上层模块,依次类推。
相对导入对于包的维护优势
相对导入可以避免硬编码带来的包维护问题,例如我们改了某一层包的名称,那么其它模块对于其子包的所有绝对导入就不能用了,但是采用相对导入语句的模块,就会避免这个问题。
需要注意:存在相对导入语句的模块,是不能直接运行的。 例如,对于如下层次结构的 Digital.py 文件,
# \PHONE # │ common_util.py -> setup() # │ __init__.py # │ # ├─Fax # │ G3.py -> bar() # │ __init__.py # │ # ├─Mobile # │ Analog.py -> foo() # │ Digital.py # │ __init__.py # │ # ├─Pager # │ Page.py # │ __init__.py # │ # └─Voice # Isdn.py # __init__.py # from .Analog import foo # ValueError: Attempted relative import in non-package from ..common_util import setup # ValueError: Attempted relative import in non-package from ..Fax.G3 import bar # ValueError: Attempted relative import in non-package if __name__ == '__main__': foo() setup() bar()
如果上述代码直接运行,将导致 ValueError 异常,
ValueError: Attempted relative import in non-package
这是因为:一个模块直接运行,Python 认为这个模块就是顶层模块,不存在层次结构,所以找不到其它的相对路径。
而要正确运行,就要显式的指定路径,如下,
C:\workspace\X_python>python -m Phone.Mobile.Digital This is foo() from Phone.Mobile.Analog This is setup() from Phone.common_util This is bar() from Phone.Fax.G3
神龙|纯净稳定代理IP免费测试>>>>>>>>天启|企业级代理IP免费测试>>>>>>>>IPIPGO|全球住宅代理IP免费测试