為什么 Python 沒有 main 函數(shù)?
“
閱讀本文大概需要 3 分鐘。

眾所周知,Python 中沒有所謂的 main 函數(shù),但是網(wǎng)上經(jīng)常有文章提到“ Python 的 main 函數(shù)”和“建議編寫 main 函數(shù)”。
其實(shí),可能他們是想模仿真正的 main 函數(shù),但是許多人都被誤導(dǎo)(或誤解),然后編寫了非常笨拙的代碼。
在開始討論之前,我們先來回答以下兩個問題:
-
所謂的“main 函數(shù)”究竟是什么意思?
-
為什么有些編程語言必須編寫main函數(shù)?
一些編程語言將 main 函數(shù)作為程序的執(zhí)行入口,比如 C/C++、C#、Java、Go、Rust 等等,這個函數(shù)具有特定的含義:
-
main 函數(shù)名是必須的,這意味著必須有一個主函數(shù)。
-
最多只能有一個 main 函數(shù),這意味著程序的入口是唯一的。
-
語法格式有特定要求,書寫形式也相對固定。
為什么必須強(qiáng)制 main 函數(shù)作為入口?
這些語言都是編譯語言,需要將代碼編譯成可執(zhí)行的二進(jìn)制文件。為了讓操作系統(tǒng)/引導(dǎo)程序找到程序的開頭,需要定義這樣一個函數(shù)。
簡而言之,需要在大量可執(zhí)行的代碼中定義一個至關(guān)重要的的開頭。
不難看出,對于這些語言來說,main 函數(shù)是不可或缺的組成部分。
但是,當(dāng)我們把目光轉(zhuǎn)向 Python 時(shí),就會發(fā)現(xiàn)情況大不相同。
-
Python 是一種解釋語言,即腳本語言。運(yùn)行過程是從上到下,逐行進(jìn)行的,這意味著它的起點(diǎn)是已知的。
-
每個 .py 文件都是一個可執(zhí)行文件,可作為整個程序的入口文件,意味著該程序的入口很靈活,而且無需遵循任何約定。
-
有時(shí)運(yùn)行 Python 項(xiàng)目時(shí)不需要有指定入口文件(命令行比較常見,例如“ python -m http.server 8000”),可能是因?yàn)樵擁?xiàng)目中有 main.py 文件,在軟件包中作為“文件”來執(zhí)行。
總而言之,Python 作為腳本語言不同于編譯語言。無論是單個模塊(即 .py 文件),還是由多個模塊組成的軟件包,Python 都可以選擇一種靈活的執(zhí)行方法,這完全不像其他語言那樣必須定義入口。
換句話說,Python 不需要規(guī)定程序員必須在語法上定義一個統(tǒng)一的入口(無論是函數(shù)、類還是其他東西)。
有些學(xué)生可能會感到困惑,因?yàn)樗麄兘?jīng)??吹交蚓帉懸韵麓a:
#?main?file
def?main():????
????……
if?__name__?==?'__main__':????
????main()
這不就是 Python 的 main 函數(shù)嗎?相信很多人都這么認(rèn)為!
不,并不是。
除了函數(shù)名是“main”之外,這段代碼與我們前面介紹的 main 函數(shù)沒有半點(diǎn)關(guān)系,這個函數(shù)既不是必須的,也不能確定程序的執(zhí)行順序。即便沒有上面這樣的 main 函數(shù),也不會有任何的語法問題。
人們想編寫一個 main 函數(shù)的主要原因其實(shí)是為了強(qiáng)調(diào)這是一個主函數(shù),希望人為地將其設(shè)置成第一個執(zhí)行的函數(shù)。
他們可能認(rèn)為這個名字的函數(shù)更容易記住。
他們之所以要編寫?name?=='main',可能是因?yàn)橄氡砻?main() 只在直接執(zhí)行當(dāng)前腳本時(shí)才運(yùn)行,而在將其導(dǎo)入到其他模塊時(shí)不要運(yùn)行。
但是,我個人不推薦這種寫法。
舉一個簡單的例子,假設(shè)只有幾十行代碼,或者一個腳本文件實(shí)現(xiàn)了一個簡單的功能(一個爬蟲,或畫一只烏龜,等等),但都是按照前面的方式編寫的。
不推薦 if?name?== 'main' 的寫法,因?yàn)椋?/p>
-
首先,如果只有一個文件的話,那么這個文件不可能被導(dǎo)出。
-
其次,如果有多個文件,強(qiáng)烈建議不要將這個語句寫在入口文件(main.py)中。從理論上講,它的內(nèi)容不應(yīng)該導(dǎo)出供其他模塊使用,因?yàn)樗瞧瘘c(diǎn)。
-
最后,在多個文件的情況下,也不建議在非入口的文件中寫入這條語句,因?yàn)檫@條語句能做的最多也就是編寫一些測試代碼。即便如此,測試代碼也應(yīng)分開寫到專用目錄或文件中。
每次看到這些笨拙的代碼時(shí),我都會感到不適。為什么要寫這樣的 if 語句?你壓根不應(yīng)該將這段代碼包裝成一個函數(shù)!
總結(jié)
-
打破慣性思維,編寫真實(shí)的代碼。main 函數(shù)是某些語言的唯一入口,但不應(yīng)在 Python 中使用。你應(yīng)該了解腳本語言的特征,并學(xué)習(xí)簡單而優(yōu)雅的風(fēng)格。
-
你可以使用 main.py,而不是編寫 main 函數(shù)。由于 Python 程序的執(zhí)行單元是腳本文件,而不是函數(shù)或類,因此建議將入口文件命名為 main.py,并根據(jù)需要決定內(nèi)部的函數(shù)。
-
將 main.py 作為入口文件。該文件可直接與命令行的 “-m” 參數(shù)結(jié)合使用。

End
崔慶才的新書《Python3網(wǎng)絡(luò)爬蟲開發(fā)實(shí)戰(zhàn)(第二版)》已經(jīng)正式上市了!書中詳細(xì)介紹了零基礎(chǔ)用 Python 開發(fā)爬蟲的各方面知識,同時(shí)相比第一版新增了 JavaScript 逆向、Android 逆向、異步爬蟲、深度學(xué)習(xí)、Kubernetes 相關(guān)內(nèi)容,?同時(shí)本書已經(jīng)獲得 Python 之父 Guido 的推薦,目前本書正在七折促銷中!
內(nèi)容介紹:《Python3網(wǎng)絡(luò)爬蟲開發(fā)實(shí)戰(zhàn)(第二版)》內(nèi)容介紹

掃碼購買
