為什么我不推薦在Python里寫main函數(shù)?
學(xué)過(guò)一陣子 Python 的同學(xué)應(yīng)該都知道,Python 中沒有所謂的 main 入口函數(shù),但是網(wǎng)上經(jīng)常看到一些文章提“Python 的 main 函數(shù)”、“建議寫 main 函數(shù)”……
有些人是知情的,他的意圖可能是模仿那些正宗的 main 函數(shù),但還有不少人明顯是被誤導(dǎo)了(或自己誤解了),就寫出來(lái)很累贅的代碼。
本期“Python 為什么”欄目來(lái)聊聊 Python 為什么沒有 main 函數(shù)?
在開始正題之前,先要來(lái)回答這兩個(gè)問(wèn)題:所謂的 “main 函數(shù)”是指什么?為什么有些編程語(yǔ)言需要強(qiáng)制寫一個(gè) main 函數(shù)?
main 函數(shù)名是強(qiáng)制的,也就是要求必須有一個(gè) main 函數(shù)
main 函數(shù)最多只能有一個(gè),也就是說(shuō)程序的入口是唯一的
語(yǔ)法格式有一定的要求,具有相對(duì)固定的模板
這些語(yǔ)言是編譯型語(yǔ)言,需要把代碼編譯成可執(zhí)行的二進(jìn)制文件,為了讓操作系統(tǒng)/啟動(dòng)器找到程序的起點(diǎn),所以要約定這一個(gè)函數(shù)。簡(jiǎn)單地說(shuō),就是在一大堆代碼里,需要定義一個(gè)顯著的可用于執(zhí)行的開頭。
不難看出,main 函數(shù)是那些語(yǔ)言中重要而不可缺的有機(jī)組成部分。
Python 是解釋型語(yǔ)言,即腳本語(yǔ)言,運(yùn)行過(guò)程是從上往下,逐行解析運(yùn)行,也就是說(shuō)它的起點(diǎn)是可知的
每個(gè) .py 文件就是一個(gè)可執(zhí)行文件,都可作為整個(gè)程序的入口文件,也就是說(shuō)程序的入口是靈活可變的,沒有必須遵守的約定
有時(shí)候運(yùn)行 Python 項(xiàng)目,并沒有指定入口文件(命令行中較常見,例如"python -m http.server 8000"), 那可能是存在
__main__.py文件,它所在的包被當(dāng)成一個(gè)“文件”來(lái)執(zhí)行了
也就是說(shuō),Python 沒有必要在語(yǔ)法層面規(guī)定程序員必須定義出一個(gè)統(tǒng)一的入口(不管是函數(shù)還是類還是什么東西)。
有些同學(xué)可能會(huì)有疑惑,因?yàn)樗麄兘?jīng)常看到或者自己寫出下面這樣的代碼:
# main 里是某些主體代碼
def main():
……
if __name__ == '__main__':
main()
難道這不就是 Python 的 main 函數(shù)么?相信有不少同學(xué)會(huì)這么想!
非也!非也!
除了函數(shù)名是“main”以外,它跟我們前面介紹的正統(tǒng)的 main 函數(shù)沒有半毛錢關(guān)系,既沒有強(qiáng)制性,也沒有必然決定程序執(zhí)行順序的作用。缺少它,也不會(huì)導(dǎo)致什么語(yǔ)法問(wèn)題。
之所以有些知情人要命名出一個(gè)”main“函數(shù),其實(shí)是想強(qiáng)調(diào)它的”主要“地位,想要人為地安排它作為第一個(gè)執(zhí)行的函數(shù)。他們可能認(rèn)為這樣命名的函數(shù),比較容易記憶。
之所以有些知情人要寫if __name__ == '__main__' ,可能想表明 main() 只有在當(dāng)前腳本被直接執(zhí)行時(shí)才運(yùn)行,不希望被導(dǎo)入其它模塊時(shí)運(yùn)行。
對(duì)于這些“知情人”,他們有一定的道理。
但是,我個(gè)人并不推薦這種寫法,甚至有時(shí)候會(huì)非常反感!
最明顯的例子:明明只有幾十行代碼,或者僅有一個(gè)腳本文件,實(shí)現(xiàn)一個(gè)簡(jiǎn)單的功能(一小段爬蟲、用 turtle 畫張圖等等),但是它們都按前面的樣式寫了。
我每次看到這種不假思索的累贅代碼,就覺得難受。為什么要寫那行 if 語(yǔ)句呢?可能的話,應(yīng)該拆分 main 函數(shù),甚至不必封裝成一個(gè)函數(shù)??!
打破慣性思維,寫出地道的代碼。main 入口函數(shù)是某些語(yǔ)言特有的,不該在 Python 中“照貓畫虎”,應(yīng)該了解腳本語(yǔ)言的特點(diǎn),寫出簡(jiǎn)潔優(yōu)雅的風(fēng)格
使用 main.py 而非 main()。因?yàn)?Python 的程序執(zhí)行單位其實(shí)是腳本文件,而非某個(gè)函數(shù)或者類,所以建議把入口文件命名為 main.py,內(nèi)部的函數(shù)按需求而定
可以的話,使用
__main__.py作為入口文件。這個(gè)文件結(jié)合命令行的“-m”參數(shù)使用,非常好用。不推薦寫
if __name__ == '__main__'。首先,如果只有一個(gè)文件的話,因?yàn)椴淮嬖趯?dǎo)出的可能,不建議寫。其次,存在多文件時(shí),入口文件(main.py)中極不推薦寫這一句,此文件的代碼邏輯應(yīng)該精煉,理論上其內(nèi)容不該被導(dǎo)出到其它模塊使用,因?yàn)樗瞧瘘c(diǎn)!最后,多文件的非入口文件也不建議寫,因?yàn)樵诜侨肟谖募袑戇@個(gè)判斷,最大的作用就是寫一些測(cè)試代碼,但是測(cè)試代碼應(yīng)該分離出來(lái),寫到專門的目錄或文件中。
_往期文章推薦_
