Python的包导入机制
- 2020 年 1 月 7 日
- 筆記
Background
在Python的大型项目中,一般都会用到模块包来组织文件层次,其中当一个目录内含有__init__ . py文件时,就可以視该目录为一个模块包。 当在模块包中使用import语句的时候,不同的语法会导致不同的模块搜索导入方式,常见的导入方式如下:
- 绝对导入(absolute import)
- 显式相对导入(explicit relative import)
- 隐式相对导入(implicit relative import)
需要注意的是,这些导入方式都是对于模块包而言,对于一般的模块还是从sys.path搜索入手。
Prerequisite
- Python脚本运行的方式?
- Python脚本运行分为两种方式:一种是作为top level script运行,另一种则是作为被导入的包模块运行。
- 当使用python命令直接执行一个py文件的时候,该文件就是以top level script方式运行,此时文件的__name__属性则为__main__。 # test.py print __name__ # output __main__
- 当文件使用包模块的方式运行的时候,文件的__name__属性则为模块的路径(从top level script的目录开始),包模块的例子目录结构如下: ├── main.py └── pac ├── __init__.py └── moduleA.py 笔者将直接运行main.py文件,代码如下: ## main.py from pac import moduleA ## moduleA.py print __name__ ## output pac.moduleA
- 这两种运行方式还有不同的地方是,当使用top level script的方式运行的时候是不会生成字节码的(即.pyc文件),而通过包模块的方式则会生成字节码。
Relative Import And Absolute Import
假设如下的import语句:
import string
这个string是当前目录下的string模块呢,还是在标准库的string模块呢?在早期的Python中,当使用import语句的时候,都会优先寻找目录内的模块,因此这就是隐式相对导入。
但是在有同名模块的情况下,如果还想引用标准库中的string模块那该怎么办?因此Python实现了绝对导入,在绝对导入的模式下,当使用import string的时候,就会优先搜索当前目录以外的模块。绝对导入模式是Python3默认采取的包导入方式,其实这种方式在Python2.5及以上版本就已经实现,要想使用只需加上:
from __future__ import absolute_import
关于隐式相对导入于绝对导入的例子如下: 包结构:
├── main.py └── pac ├── __init__.py ├── __init__.pyc ├── explicit_import.py ├── explicit_import.pyc ├── implicit_import.py ├── implicit_import.pyc ├── string.py └── string.pyc
代码如下:
# main.py from pac import implicit_import from pac import explicit_import # explicit_import.py from __future__ import absolute_import import string print string.digits # implicit_import.py import string print string.digits # string.py digits = '2333' ## output 2333(relative import) 0123456789(absolute import)
绝对导入还有一种使用方法,比如在explicit_import.py中可以通过:
from pac.implicit_import import *
来引用implicit_import文件中的变量。
explicit relative import
虽然绝对导入能够完成相对导入的所有功能,但是显式的相对导入也是可以接受的。当使用.语法的时候就是使用相对导入:
# 导入当前目录下的string模块 # right from . import string # wrong import .string
至于下面的导入方法错误的原因,这是因为Python语法不支持的缘故。 同时值得注意的是,显式的相对导入是根据模块的__name__属性来确定相对位置的,因此如果是在top level script中,显式相对导入并不能使用,会报出如下错误:
ValueError: Attempted relative import in non-package
当然,在PEP 366 – Main module explicit relative imports中,也给出了在Python中执行非包内的模块(作为top level脚本执行)使用显示相对导入的方法:在执行python命令时加上-m选项,此时就会启用模块的__package__属性。
详细的关于相对导入与绝对导入参考:PEP 328 – Imports: Multi-Line and Absolute/Relative。