python import 機制初探

引子

考慮有以下代碼結構html

.
├── cat
│   ├── __init__.py
│   ├── cat.py
│   └── moo.py
└── dog
    ├── __init__.py
    └── dog.py

2 directories, 5 files

狀況1:python

# ./cat/cat.py
import moo

其他文件爲空。若是咱們用運行命令: python cat/cat.py,則程序正常運行;若是運行命令;python -m cat.cat,則會報錯;ide

狀況2:命令行

# ./cat/cat.py
from . import moo

則運行結果相反。code

其實,咱們的問題早就有人提出過了。PEP 395orm

The fact that most methods of invoking Python code from the command line break when that code is inside a package, and the two that do work are highly sensitive to the current working directory is all thoroughly confusing for a beginner. :若是一段代碼在一個包內,那麼從命令行使用這段代碼是很是很是容易出錯的,只有2種方法可行:
  1. 在整個包的外部,python -c "from module import code
  2. 在整個包的外部,經過-m參數執行。

若是在包內部,即便你的代碼正確,也基本會出錯。htm

分析

狀況1:ci

  • 當咱們運行python cat/cat.py時,這時候系統的搜素路徑包括./cat/,咱們能夠經過在./cat/cat.py中添加import sys; print(sys.path)證明這一點。
  • 當咱們運行 python -m cat.cat時,系統的搜索路徑只包括.,也就是當前文件夾,咱們能夠經過在當前文件夾下添加一個moo.py文件,這樣python -m cat.cat就不會報錯了。可見,這時的搜索空間,不包括自文件夾。

狀況2:文檔

狀況2其實和狀況1很不同,狀況2叫作package relative import,也就是包內間接引用。顧名思義,這種引用方法只能在包內使用。也就是說,只能經過python -m cat.cat, 或者是python -c "from cat import cat" 之類的方法來使用。如下內容源自python官方文檔中的 The import statementget

When a module or package is contained within another package ... using leading dots in the specified module or package after from you can specify how high to traverse up the current package hierarchy without specifying exact names.

官方文檔中的Package Relative Import 展現了詳細的用法。

import

python中,載入一個包有幾種方式:

  1. import語句
  2. importlib模塊提供的方法
  3. 內置方法__import__ (不提倡)

python中,有module,有packagepackage都是module, 但module不必定是packagemodule只有一種,package有2種。

代碼重載:importlib.reload

使用importlib.reload會重載當前代碼。

import os
import importlib

import foo
print("foo.a: {}".format(foo.a))

command = 'echo a=2 >> foo.py'

print(command)
os.popen(command)

importlib.reload(foo)
print("foo.a: {}".format(foo.a))

輸出:

foo.a: 2
echo a=2 >> foo.py
foo.a: 2

參考資料

import 語法

PEP 395

PEP 420

相關文章
相關標籤/搜索