Python 沒有main函數(shù)的原因
毫無疑問 Python 中沒有所謂的 main 入口函數(shù),但是網(wǎng)上經(jīng)常看到一些文章提“Python 的 main 函數(shù)”、“建議寫 main 函數(shù)”……
有些人是知情的,他的意圖可能是模仿那些正宗的 main 函數(shù),但還有不少人明顯是被誤導(dǎo)了(或自己誤解了),就寫出來很累贅的代碼。
本期“Python 為什么”欄目來聊聊 Python 為什么沒有 main 函數(shù)?
在開始正題之前,先要來回答這兩個(gè)問題:所謂的 “main 函數(shù)”是指什么?為什么有些編程語言需要強(qiáng)制寫一個(gè) main 函數(shù)?
某些編程語言以 main 函數(shù)作為程序的執(zhí)行入口,例如 C/C++、C#、 Java、Go 和 Rust 等,它們具有特定的含義:
- main 函數(shù)名是強(qiáng)制的,也就是要求必須有一個(gè) main 函數(shù)
- main 函數(shù)最多只能有一個(gè),也就是說程序的入口是唯一的
- 語法格式有一定的要求,具有相對固定的模板
為什么要強(qiáng)制一個(gè) main 入口函數(shù)呢?
這些語言是編譯型語言,需要把代碼編譯成可執(zhí)行的二進(jìn)制文件,為了讓操作系統(tǒng)/啟動(dòng)器找到程序的起點(diǎn),所以要約定這一個(gè)函數(shù)。簡單地說,就是在一大堆代碼里,需要定義一個(gè)顯著的可用于執(zhí)行的開頭。
不難看出,main 函數(shù)是那些語言中重要而不可缺的有機(jī)組成部分。
然而,我們再來看看 Python,情況就大不相同了。
- Python 是解釋型語言,即腳本語言,運(yùn)行過程是從上往下,逐行解析運(yùn)行,也就是說它的起點(diǎn)是可知的
- 每個(gè) .py 文件就是一個(gè)可執(zhí)行文件,都可作為整個(gè)程序的入口文件,也就是說程序的入口是靈活可變的,沒有必須遵守的約定
- 有時(shí)候運(yùn)行 Python 項(xiàng)目,并沒有指定入口文件(命令行中較常見,例如"python -m http.server 8000"), 那可能是存在
__main__.py
文件,它所在的包被當(dāng)成一個(gè)“文件”來執(zhí)行了
歸結(jié)起來,意思是說 Python 這種腳本語言跟編譯型語言不同,它不管是在單個(gè)模塊層面(即一個(gè) .py 文件),還是在由多個(gè)模塊組成的包層面,都可選擇靈活的執(zhí)行方式,不像其它語言缺了約定好的入口就沒法執(zhí)行。
也就是說,Python 沒有必要在語法層面規(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)致什么語法問題。
之所以有些知情人要命名出一個(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)行。
對于這些“知情人”,他們有一定的道理。
但是,我個(gè)人并不推薦這種寫法,甚至有時(shí)候會(huì)非常反感!
最明顯的例子:明明只有幾十行代碼,或者僅有一個(gè)腳本文件,實(shí)現(xiàn)一個(gè)簡單的功能(一小段爬蟲、用 turtle 畫張圖等等),但是它們都按前面的樣式寫了。
我每次看到這種不假思索的累贅代碼,就覺得難受。為什么要寫那行 if 語句呢?可能的話,應(yīng)該拆分 main 函數(shù),甚至不必封裝成一個(gè)函數(shù)?。?/p>
我個(gè)人總結(jié)出以下的經(jīng)驗(yàn):
- 打破慣性思維,寫出地道的代碼。main 入口函數(shù)是某些語言特有的,不該在 Python 中“照貓畫虎”,應(yīng)該了解腳本語言的特點(diǎ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ù)使用,非常好用。推薦閱讀:為什么是 Python -m - 不推薦寫
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è)判斷,最大的作用就是寫一些測試代碼,但是測試代碼應(yīng)該分離出來,寫到專門的目錄或文件中。
小結(jié):本文首先解釋了什么是 main 入口函數(shù),以及為什么某些語言會(huì)強(qiáng)制要求寫 main 函數(shù);接著,解釋了為什么 Python 不需要寫 main 函數(shù);最后則是針對某些人存在的慣性誤區(qū),分享了我個(gè)人的四點(diǎn)編程經(jīng)驗(yàn)。
以上就是Python 沒有main函數(shù)的原因的詳細(xì)內(nèi)容,更多關(guān)于Python main函數(shù)的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
python 3.5實(shí)現(xiàn)檢測路由器流量并寫入txt的方法實(shí)例
這篇文章主要給大家介紹了關(guān)于利用python 3.5實(shí)現(xiàn)檢測路由器流量并寫入txt的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起看看吧。2017-12-12Python常用字符串替換函數(shù)strip、replace及sub用法示例
這篇文章主要介紹了Python常用字符串替換函數(shù)strip、replace及sub用法,結(jié)合實(shí)例形式分析了Python針對字符串替換的常用函數(shù)strip、replace及sub功能及簡單使用技巧,需要的朋友可以參考下2018-05-05python3 設(shè)置多進(jìn)程名稱并在ps命令中可見(Centos7 系統(tǒng))
setproctitle 是一個(gè) Python 模塊,用于設(shè)置進(jìn)程標(biāo)題(process title),通過設(shè)置進(jìn)程標(biāo)題,可以讓進(jìn)程在系統(tǒng)級的進(jìn)程管理工具中展示自定義的名稱,方便用戶查看和管理進(jìn)程,本文介紹python3 設(shè)置多進(jìn)程名稱并在ps命令中可見,感興趣的朋友一起看看吧2024-03-03Python實(shí)現(xiàn)在PyPI上發(fā)布自定義軟件包的方法詳解
在Python中我們經(jīng)常使用pip來安裝第三方Python軟件包,其實(shí)我們每個(gè)人都可以免費(fèi)地將自己寫的Python包發(fā)布到PyPI上。本文我們就將詳細(xì)介紹如何發(fā)布測試包,需要的可以參考一下2022-06-06Django框架ORM操作數(shù)據(jù)庫不生效問題示例解決方法
本文詳細(xì)描述使用Django 的ORM框架操作PostgreSQL數(shù)據(jù)庫刪除不生效問題的定位過程及解決方案,并總結(jié)使用ORM框架操作數(shù)據(jù)庫不生效的問題的通用定位方法,感興趣的朋友跟隨小編一起看看吧2023-01-01關(guān)于TensorBoard可視化不顯示數(shù)據(jù)問題No scalar data was&nbs
這篇文章主要介紹了如何解決TensorBoard可視化不顯示數(shù)據(jù)問題No scalar data was found,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09