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