Python的包导入机制

2023-01-31 05:01:00 python 导入 机制

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。

相关文章