在剛?cè)腴Tpython時(shí),模塊化編程、模塊、類庫(kù)等術(shù)語常常并不容易理清。尤其是Modules(模塊)和Packages(包),在import引用時(shí)很容易混淆出錯(cuò)。
實(shí)際上,Python中的函數(shù)(Function)、類(Class)、模塊(Module)、包庫(kù)(Package),都是為了實(shí)現(xiàn)模塊化引用,讓程序的組織更清晰有條理。
Modules2模塊是包含 Python 定義和語句的文件。以.py為后綴的文件名就是模塊名稱。
在模塊內(nèi),模塊的名稱可以用全局變量 __name__表示(字符串)。
舉個(gè)例子,我們創(chuàng)建了一個(gè)fibo.py文件,文件內(nèi)容為:
# Fibonacci numbers module def fib(n): # write Fibonacci series up to n a, b = 0, 1 while a n: print(a, end=' ') a, b = b, a+b print() def fib2(n): # return Fibonacci series up to n result = [] a, b = 0, 1 while a n: result.append(a) a, b = b, a+b return result
這里fibo.py就是一個(gè)模塊,fib、fib2是fibo模塊中的函數(shù)。
如果我們想要在其他程序中使用fibo模塊,可以有以下三種方式導(dǎo)入:
#import module_name import fibo
可使用下面的語法來使用其中任何一個(gè)函數(shù):
#module_name.func() fibo.fib(10)
⭕ 注意:這里模塊名+句點(diǎn)不可省略。
#from module_name import function_name from fibo import fib, fib2 fib(10)
若使用這種語法,調(diào)用函數(shù)時(shí)就無需使用模塊名+句點(diǎn)。
因?yàn)樵趇mport語句中已經(jīng)顯式地導(dǎo)入了函數(shù)fib和fib2,因此調(diào)用它時(shí)只需指定其名稱。
#from module_name import * from fibo import * fib(20)
這種方式會(huì)導(dǎo)入除可下劃線 (__)開頭的名稱以外的所有函數(shù)。
⭕ 注意:在大多數(shù)情況下,通常不推薦*這種用法,因?yàn)樗赡軙?huì)在解釋器中引入了一組未知的名稱,而且通常會(huì)導(dǎo)致代碼可讀性變差。
# import module as m import numpy as np a = np.arange(100)
利用as給導(dǎo)入模塊一個(gè)別名,簡(jiǎn)化代碼中的調(diào)用寫法。
如果我們想單獨(dú)測(cè)試下模塊,可以在模塊中添加以下代碼,就可以既用作腳本,也可用作可導(dǎo)入模塊:
if __name__ == "__main__": import sys fib(int(sys.argv[1]))
單獨(dú)運(yùn)行模塊:
python fibo.py 100
這段解析命令行的代碼僅在模塊作為“主”文件執(zhí)行時(shí)才運(yùn)行。
為了加快模塊的加載速度,Python 會(huì)將每個(gè)模塊的編譯版本(如*.pyc)會(huì)緩存在__pycache__下的目錄中。生成編譯文件pyc的詳細(xì)過程,可以參考文檔PEP 3147。
Python 會(huì)根據(jù)編譯版本檢查源代碼的修改日期,以查看它是否已過期并需要重新編譯。
Packages包可以理解為一組模塊的容器,并用Package.Module的方式來構(gòu)建命名空間3。
以文件系統(tǒng)來類比的話,可以將包視為文件系統(tǒng)上的目錄,而將模塊視為目錄中的文件。4
例如,A.B指定的是在命名為B的包中命名為A的子模塊。
利用這樣的方法,可以避免一些多模塊的包之間命名發(fā)生沖突的問題,有點(diǎn)類似于C++中的std::string、cv::imread等命名空間的引用。
例如,這是一個(gè)官方的package例子,提供了關(guān)于聲音處理的sound包:
sound/ Top-level package __init__.py Initialize the sound package formats/ Subpackage for file format conversions __init__.py wavread.py wavwrite.py ... effects/ Subpackage for sound effects __init__.py echo.py ... filters/ Subpackage for filters __init__.py equalizer.py ...
from sound.effects import echo echo.echofilter(input, output, delay=0.7, atten=4)
這種方式,可以直接引用函數(shù),而不用加上前面包的前綴。
from sound.effects.echo import echofilter echofilter(input, output, delay=0.7, atten=4)
這種方式會(huì)加載子模塊echo,同時(shí)使子模塊中的 echofilter() 函數(shù)直接可用。
from package import item語句會(huì)先測(cè)試item是否在包中定義;如果包中沒有找到定義,會(huì)假定item是一個(gè)模塊并嘗試加載它。如果依舊找不到item, 就會(huì)引發(fā)我們常見的ImportError異常。
from . import echo from .. import formats from ..filters import equalizer
這里的.可以訪問同級(jí)目錄下的包(Package)或者模塊(Module)。
這里的..可以訪問上一級(jí)目錄下的包(Package)或者模塊(Module)。
當(dāng)我們直接采用from sound.effects import *時(shí),可能會(huì)引用一些不需要的內(nèi)容,或者導(dǎo)致加載速度過慢。
這時(shí)我們可以通過在__init__.py中定義一個(gè)_all__列表,來指定用 * 時(shí)應(yīng)導(dǎo)入的模塊名稱列表:
__all__ = ["echo", "surround", "reverse"]
這樣我們就可以維護(hù)在import * 時(shí)需要導(dǎo)入的模塊列表,在發(fā)布不同版本的包時(shí)很有用。
通過setuptool工具打包自己的Package,可以參考這兩篇文檔:
https://packaging.python.org/tutorials/packaging-projects/
如何添加必要的文件和結(jié)構(gòu)來創(chuàng)建包、如何構(gòu)建包以及如何將其上傳到Python Package Index。
https://packaging.python.org/guides/distributing-packages-using-setuptools/
介紹如何配置、打包和分發(fā)自己的 Python 項(xiàng)目的基礎(chǔ)知識(shí)。
注意,在python社區(qū)中的Package,通常是指發(fā)行版的軟件包,而不是源代碼中一組模塊的容器(a container of modules)。
常見的包可以查看PyPI:https://pypi.org/,通過使用pip install來安裝社區(qū)提供的Packages。
模塊化就是將相關(guān)代碼組織到不同層級(jí)的文件里,便于復(fù)用,提高代碼的可讀性。
函數(shù)、變量、類存儲(chǔ)在被稱為模塊(Module)的.py文件中,一組模塊文件又組成了包(Package)。
引入Package包或Module模塊,可以用import …或from … import …,還可以通過相對(duì)路徑引用上一級(jí)的包和模塊。
本篇文章就到這里了,希望能給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
標(biāo)簽:漯河 南平 林芝 大同 海南 盤錦 寧夏 普洱
巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《詳解Python模塊化--模塊(Modules)和包(Packages)》,本文關(guān)鍵詞 詳解,Python,模塊化,模塊,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。