基本概念

Python 中的包,即包含 __init__.py 文件的文件夹。

对于 Python 的包内导入,即包内模块导入包内模块,存在绝对导入和相对导入问题。

普通 Python 模块的搜索路径

1. 在当前模块所在路径中搜索导入模块

2. 在环境变量 PYTHONPATH 指定的路径列表中搜索导入模块

3. 在 sys.path 指定的路径列表中搜索导入模块

Python import 的步骤

Python 所有加载的模块信息都存放在 sys.modules 字典结构中,当 import 一个模块时,会按如下步骤来进行

1. 如果 import A,检查 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 文件,

  1. #!/usr/bin/env python
  2. # -*- coding: utf-8 -*-
  3. ##############################################################################
  4. # Purpose: to demo underlayer import upperlayer.
  5. ##############################################################################
  6. #
  7. # \PHONE
  8. # │ common_util.py -> setup()
  9. # │ __init__.py
  10. # │
  11. # ├─Fax
  12. # │ G3.py -> bar()
  13. # │ __init__.py
  14. # │
  15. # ├─Mobile
  16. # │ Analog.py -> foo()
  17. # │ Digital.py
  18. # │ __init__.py
  19. # │
  20. # ├─Pager
  21. # │ Page.py
  22. # │ __init__.py
  23. # │
  24. # └─Voice
  25. # Isdn.py
  26. # __init__.py
  27. #
  28. ##############################################################################
  29.  
  30. from .Analog import foo # ValueError: Attempted relative import in non-package
  31. from ..common_util import setup # ValueError: Attempted relative import in non-package
  32. from ..Fax.G3 import bar # ValueError: Attempted relative import in non-package
  33.  
  34. if __name__ == '__main__':
  35.  
  36. foo()
  37. setup()
  38. bar()

如果上述代码直接运行,将导致 ValueError 异常,

  1. ValueError: Attempted relative import in non-package

这是因为:一个模块直接运行,Python 认为这个模块就是顶层模块,不存在层次结构,所以找不到其它的相对路径。

而要正确运行,就要显式的指定路径,如下,

  1. C:\workspace\X_python>python -m Phone.Mobile.Digital
  2. This is foo() from Phone.Mobile.Analog
  3. This is setup() from Phone.common_util
  4. This is bar() from Phone.Fax.G3

当然,我们一般不会直接运行包内的某个模块,这里只是做个说明。

绝对导入对于包维护的劣势

例如,对于如下层次结构的 Digital.py 文件,

  1. #!/usr/bin/env python
  2. # -*- coding: utf-8 -*-
  3. ##############################################################################
  4. # Purpose: to demo underlayer import upperlayer.
  5. ##############################################################################
  6. #
  7. # \PHONE
  8. # │ common_util.py -> setup()
  9. # │ __init__.py
  10. # │
  11. # ├─Fax
  12. # │ G3.py -> bar()
  13. # │ __init__.py
  14. # │
  15. # ├─Mobile
  16. # │ Analog.py -> foo()
  17. # │ Digital.py
  18. # │ __init__.py
  19. # │
  20. # ├─Pager
  21. # │ Page.py
  22. # │ __init__.py
  23. # │
  24. # └─Voice
  25. # Isdn.py
  26. # __init__.py
  27. #
  28. ##############################################################################
  29.  
  30. # from .Analog import foo # ValueError: Attempted relative import in non-package
  31. # from ..common_util import setup # ValueError: Attempted relative import in non-package
  32. # from ..Fax.G3 import bar # ValueError: Attempted relative import in non-package
  33.  
  34. from Phone.Mobile.Analog import foo
  35. from Phone.common_util import setup
  36. from Phone.Fax.G3 import bar
  37.  
  38. if __name__ == '__main__':
  39.  
  40. foo()
  41. setup()
  42. bar()

上述代码可以直接运行。
但是,绝对导入的硬编码模式,如果在包中存在很多 Digital.py 类似模块,都采用了 from Phone.common_util import setup 的语句,如果有一天要更改 common_util 包(文件夹)的名字,那么会影响所有相关的代码。而采用相对导入就没有这个问题。

不过,绝对导入更清晰,如果包不是特别复杂,不是特别易变,那么还是建议采用绝对导入。(个人观点,仅供参考)

再举一个包内导入的例子,目录结构为,

  1. # myabc/
  2. # ├── abc.py
  3. # ├── __init__.py
  4. # └── xyz.py
  5.  
  6. # abc.py
  7.  
  8. def foo():
  9. print("This is foo from local abc module!")
  10.  
  11. # xyz.py
  12.  
  13. ##########################################
  14. #import .abc # invalid (due to abc is not a package, so cannot import directly)
  15. #import . abc # invalid (reason as above)
  16. ##########################################
  17.  
  18. #from .abc import foo # valid
  19. from . abc import foo # valid
  20.  
  21. def bar():
  22. print('bar - ', end='')
  23. foo()

外部使用 myabc 包,

  1. >>> import myabc.xyz
  2. >>> myabc.xyz.bar()
  3. bar - This is foo from local abc module!
  4. >>>
  5. >>> from myabc import xyz
  6. >>> xyz.bar()
  7. bar - This is foo from local abc module!
  8. >>>
  9. >>>
  10. >>> import myabc.abc
  11. >>> myabc.abc.foo()
  12. This is foo from local abc module!
  13. >>>
  14. >>> from myabc import abc
  15. >>> abc.foo()
  16. This is foo from local abc module!

再举个例子,

  1. # myfact/
  2. # ├── factory.py
  3. # ├── __init__.py
  4. # └── xyz.py
  5.  
  6. # factory.py
  7. def foo():
  8. print("This is foo from local abc module!")
  9.  
  10. # xyz.py
  11. ####################################################################################
  12. #from factory import foo # Invalid! ModuleNotFoundError: No module named 'factory'
  13. ####################################################################################
  14.  
  15. #from myfact.factory import foo # Valid, absolute
  16. #from .factory import foo # Valud, relative
  17. from . factory import foo # Valud, relative
  1. def bar():
    print('bar - ', end='')
    foo()

外部使用 myfact 包,

  1. >>> import myfact.xyz
  2. >>>
  3. >>> myfact.xyz.bar()
  4. bar - This is foo from local abc module!

完。

Python 包内的导入问题(绝对导入和相对导入)的更多相关文章

  1. python包导入细节

    包导入格式 导入模块时除了使用模块名进行导入,还可以使用目录名进行导入.例如,在sys.path路径下,有一个dir1/dir2/mod.py模块,那么在任意位置处都可以使用下面这种方式导入这个模块. ...

  2. 【转】python包导入细节

    [转]python包导入细节 包导入格式 导入模块时除了使用模块名进行导入,还可以使用目录名进行导入.例如,在sys.path路径下,有一个dir1/dir2/mod.py模块,那么在任意位置处都可以 ...

  3. 第10.8节 Python包的导入方式详解

    一. 包导入与模块导入的关系 由于包本质上是模块,其实对模块的许多导入方法都适用于包,但由于包的结构与模块有差异,所以二者还是有些区别的: 对包的导入,实际上就是包目录下的__init__.py文件的 ...

  4. Python 包的相对导入讲解

    [Python 包的相对导入讲解] 参考:http://www.dai3.com/python-import.html

  5. python 包导入规则

    python 包导入规则,恶心了一天,终于搞清楚了 1.目录 speed data __init__.py __init__.py static templates view __init__.py ...

  6. python 包以及循环导入

    包的认识 包通过文件夹来管理一系列功能相近的模块 包:一系列模块的集合体重点:包中一定有一个专门用来管理包中所有模块的文件包名:存放一系列模块的文件夹名字包名(包对象)存放的是管理模块的那个文件的地址 ...

  7. Python 包导入

    首先我们先了解下python中寻找模块的顺序 是否是内建模块 ->主目录 ->PYTHONPATH环境变量 ->标准库 -> 首先判断这个model是否是built-in,即内 ...

  8. pycharm导入python包

    总步骤:file --> settings --> poject interpreter --> 点击加号 --> 搜索需要导入的python包 --> 选中需要导入的p ...

  9. 4.windows如何导入python包

    python链接:https://www.python.org/downloads/release/python-2715/ pip链接:https://pypi.org/project/pip/#f ...

随机推荐

  1. js获取对象的长度

    var obj= { "name": "wuqian", "sex": "famale", "hello&qu ...

  2. HDU3440 House Man

    Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission( ...

  3. 用JavaScript实现点击左侧列表右侧显示列表内容的方法

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  4. android 圆角背景

    <?xml version="1.0" encoding="utf-8"?> <shape xmlns:android="http: ...

  5. (办公)rom包

    所谓ROM包,通俗点来讲,也就是手机上的系统安装包.使用过智能手机的朋友可能都有过这样的遗憾,自己所用的官方系统虽然能够满足绝大部分的需求,但总有一些细节不是很合自己的心愿……比如说,内置的输入法不好 ...

  6. mysql数据表的基本操作:表结构操作,字段操作

    本节介绍: 表结构操作 创建数据表. 查看数据表和查看字段. 修改数据表结构 删除数据表 字段操作 新增字段. 修改字段数据类型.位置或属性. 重命名字段 删除字段 首发时间:2018-02-18  ...

  7. [20190401]关于semtimedop函数调用.txt

    [20190401]关于semtimedop函数调用.txt --//上个星期测试,链接http://blog.itpub.net/267265/viewspace-2639675/--//关于sql ...

  8. 自动化测试基础篇--Selenium文件上传send_keys

    摘自https://www.cnblogs.com/sanzangTst/p/8358165.html 文件上传是web页面上很常见的一个功能,自动化成功中操作起来却不是那么简单. 一般分两个场景:一 ...

  9. 【ABAP CDS系列】ABAP CDS中的系统信息

    公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[ABAP CDS系列]ABAP CDS中的系统 ...

  10. c/c++ 二叉排序树

    c/c++ 二叉排序树 概念: 左树的所有节点的值(包括子节点)必须小于中心节点,右树所有节点的值(包括子节点)必须大于中心节点. 不允许有值相同的节点. 二叉排序树的特点: 中序遍历后,就是从小到大 ...