引子
考慮有如下代碼結(jié)構(gòu)
.
├── cat
│?? ├── __init__.py
│?? ├── cat.py
│?? └── moo.py
└── dog
├── __init__.py
└── dog.py
2 directories, 5 files
情況1:
# ./cat/cat.py
import moo
其余文件為空。如果我們用運(yùn)行命令:
python cat/cat.py
,則程序正常運(yùn)行;如果運(yùn)行命令;
python -m cat.cat
,則會報(bào)錯(cuò);
情況2:
# ./cat/cat.py
from . import moo
則運(yùn)行結(jié)果相反。
其實(shí),我們的問題早就有人提出過了。PEP 395
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. :如果一段代碼在一個(gè)包內(nèi),那么從命令行使用這段代碼是非常非常容易出錯(cuò)的,只有2種方法可行:
-
在整個(gè)包的外部,
python -c "from module import code
, -
在整個(gè)包的外部,通過
-m
參數(shù)執(zhí)行。
如果在包內(nèi)部,即使你的代碼正確,也基本會出錯(cuò)。
分析
情況1:
-
當(dāng)我們運(yùn)行
python cat/cat.py
時(shí),這時(shí)候系統(tǒng)的搜素路徑包括./cat/
,我們可以通過在./cat/cat.py
中添加import sys; print(sys.path)
證實(shí)這一點(diǎn)。 -
當(dāng)我們運(yùn)行
python -m cat.cat
時(shí),系統(tǒng)的搜索路徑只包括.
,也就是當(dāng)前文件夾,我們可以通過在當(dāng)前文件夾下添加一個(gè)moo.py
文件,這樣python -m cat.cat
就不會報(bào)錯(cuò)了。可見,這時(shí)的搜索空間,不包括自文件夾。
情況2:
情況2其實(shí)和情況1很不一樣,情況2叫做
package relative import
,也就是包內(nèi)間接引用。顧名思義,這種引用方法
只能在包內(nèi)使用
。也就是說,只能通過
python -m cat.cat
, 或者是
python -c "from cat import cat"
之類的方法來使用。以下內(nèi)容源自
python
官方文檔中的 The import statement
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 展示了詳細(xì)的用法。
import
python
中,載入一個(gè)包有幾種方式:
-
import
語句 -
importlib
模塊提供的方法 -
內(nèi)置方法
__import__
(不提倡)
python
中,有
module
,有
package
。
package
都是
module
, 但
module
不一定是
package
。
module
只有一種,
package
有2種。
代碼重載:
importlib.reload
使用
importlib.reload
會重載當(dāng)前代碼。
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
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯(lián)系: 360901061
您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長非常感激您!手機(jī)微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
