詳細(xì)聊一聊為什么Python沒有main函數(shù)
前言
眾所周知,Python中沒有所謂的main函數(shù),但是網(wǎng)上經(jīng)常有文章提到“ Python的main函數(shù)”和“建議編寫main函數(shù)”。
其實(shí),可能他們是想模仿真正的main函數(shù),但是許多人都被誤導(dǎo)(或誤解),然后編寫了非常笨拙的代碼。
在開始討論之前,我們先來(lái)回答以下兩個(gè)問題:
- 所謂的“main函數(shù)”究竟是什么意思?
- 為什么有些編程語(yǔ)言必須編寫main函數(shù)?
一些編程語(yǔ)言將main函數(shù)作為程序的執(zhí)行入口,比如C/C++、C#、Java、Go、Rust等等,這個(gè)函數(shù)具有特定的含義:
- main函數(shù)名是必須的,這意味著必須有一個(gè)主函數(shù)。
- 最多只能有一個(gè)main函數(shù),這意味著程序的入口是唯一的。
- 語(yǔ)法格式有特定要求,書寫形式也相對(duì)固定。
為什么必須強(qiáng)制main函數(shù)作為入口?
這些語(yǔ)言都是編譯語(yǔ)言,需要將代碼編譯成可執(zhí)行的二進(jìn)制文件。為了讓操作系統(tǒng)/引導(dǎo)程序找到程序的開頭,需要定義這樣一個(gè)函數(shù)。
簡(jiǎn)而言之,需要在大量可執(zhí)行的代碼中定義一個(gè)至關(guān)重要的的開頭。
不難看出,對(duì)于這些語(yǔ)言來(lái)說(shuō),main函數(shù)是不可或缺的組成部分。
但是,當(dāng)我們把目光轉(zhuǎn)向Python時(shí),就會(huì)發(fā)現(xiàn)情況大不相同。
- Python是一種解釋語(yǔ)言,即腳本語(yǔ)言。運(yùn)行過程是從上到下,逐行進(jìn)行的,這意味著它的起點(diǎn)是已知的。
- 每個(gè).py文件都是一個(gè)可執(zhí)行文件,可作為整個(gè)程序的入口文件,意味著該程序的入口很靈活,而且無(wú)需遵循任何約定。
- 有時(shí)運(yùn)行Python項(xiàng)目時(shí)不需要有指定入口文件(命令行比較常見,例如“ python -m http.server 8000”),可能是因?yàn)樵擁?xiàng)目中有main.py文件,在軟件包中作為“文件”來(lái)執(zhí)行。
總而言之,Python作為腳本語(yǔ)言不同于編譯語(yǔ)言。無(wú)論是單個(gè)模塊(即.py文件),還是由多個(gè)模塊組成的軟件包,Python都可以選擇一種靈活的執(zhí)行方法,這完全不像其他語(yǔ)言那樣必須定義入口。
換句話說(shuō),Python不需要規(guī)定程序員必須在語(yǔ)法上定義一個(gè)統(tǒng)一的入口(無(wú)論是函數(shù)、類還是其他東西)。
有些學(xué)生可能會(huì)感到困惑,因?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)系,這個(gè)函數(shù)既不是必須的,也不能確定程序的執(zhí)行順序。即便沒有上面這樣的main函數(shù),也不會(huì)有任何的語(yǔ)法問題。
人們想編寫一個(gè)main函數(shù)的主要原因其實(shí)是為了強(qiáng)調(diào)這是一個(gè)主函數(shù),希望人為地將其設(shè)置成第一個(gè)執(zhí)行的函數(shù)。
他們可能認(rèn)為這個(gè)名字的函數(shù)更容易記住。
他們之所以要編寫name ==‘main’,可能是因?yàn)橄氡砻鱩ain()只在直接執(zhí)行當(dāng)前腳本時(shí)才運(yùn)行,而在將其導(dǎo)入到其他模塊時(shí)不要運(yùn)行。
但是,我個(gè)人不推薦這種寫法。
舉一個(gè)簡(jiǎn)單的例子,假設(shè)只有幾十行代碼,或者一個(gè)腳本文件實(shí)現(xiàn)了一個(gè)簡(jiǎn)單的功能(一個(gè)爬蟲,或畫一只烏龜,等等),但都是按照前面的方式編寫的。
不推薦if name == 'main’的寫法,因?yàn)椋?/p>
- 首先,如果只有一個(gè)文件的話,那么這個(gè)文件不可能被導(dǎo)出。
- 其次,如果有多個(gè)文件,強(qiáng)烈建議不要將這個(gè)語(yǔ)句寫在入口文件(main.py)中。從理論上講,它的內(nèi)容不應(yīng)該導(dǎo)出供其他模塊使用,因?yàn)樗瞧瘘c(diǎn)。
- 最后,在多個(gè)文件的情況下,也不建議在非入口的文件中寫入這條語(yǔ)句,因?yàn)檫@條語(yǔ)句能做的最多也就是編寫一些測(cè)試代碼。即便如此,測(cè)試代碼也應(yīng)分開寫到專用目錄或文件中。
每次看到這些笨拙的代碼時(shí),我都會(huì)感到不適。為什么要寫這樣的if語(yǔ)句?你壓根不應(yīng)該將這段代碼包裝成一個(gè)函數(shù)!
總結(jié)
- 打破慣性思維,編寫真實(shí)的代碼。main函數(shù)是某些語(yǔ)言的唯一入口,但不應(yīng)在Python中使用。你應(yīng)該了解腳本語(yǔ)言的特征,并學(xué)習(xí)簡(jiǎn)單而優(yōu)雅的風(fēng)格。
- 你可以使用main.py,而不是編寫main函數(shù)。由于Python程序的執(zhí)行單元是腳本文件,而不是函數(shù)或類,因此建議將入口文件命名為main.py,并根據(jù)需要決定內(nèi)部的函數(shù)。
- 將main.py作為入口文件。該文件可直接與命令行的“-m”參數(shù)結(jié)合使用。
到此這篇關(guān)于為什么Python沒有main函數(shù)的文章就介紹到這了,更多相關(guān)Python沒有main函數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python實(shí)現(xiàn)停車場(chǎng)管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)停車場(chǎng)管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-06-06
Python?where函數(shù)保姆級(jí)使用教程
本文主要和大家介紹了詳解Python中where()函數(shù)的用法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參,希望能幫助到大家2022-06-06
python對(duì)象及面向?qū)ο蠹夹g(shù)詳解
這篇文章主要介紹了python對(duì)象及面向?qū)ο蠹夹g(shù),結(jié)合實(shí)例形式詳細(xì)分析了Python面向?qū)ο笏婕暗念?、?duì)象、方法、屬性等概念與使用技巧,需要的朋友可以參考下2016-07-07
tensorflow的ckpt及pb模型持久化方式及轉(zhuǎn)化詳解
今天小編就為大家分享一篇tensorflow的ckpt及pb模型持久化方式及轉(zhuǎn)化詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來(lái)看看吧2020-02-02
在pandas中一次性刪除dataframe的多個(gè)列方法
下面小編就為大家分享一篇在pandas中一次性刪除dataframe的多個(gè)列方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來(lái)看看吧2018-04-04
python 中Arduino串口傳輸數(shù)據(jù)到電腦并保存至excel表格
這篇文章主要介紹了python Arduino串口傳輸數(shù)據(jù)到電腦并保存至excel表格,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-10-10
Python協(xié)程asyncio模塊的演變及高級(jí)用法
網(wǎng)上很多關(guān)于Python協(xié)程asyncio模塊的教程都是基于老版Python的, 本文將以對(duì)比方式展示新老Python版本下協(xié)程的寫法有什么不同并總結(jié)了asyncio的一些高級(jí)用法, 包括如何獲取協(xié)程任務(wù)執(zhí)行結(jié)果,gather和wait方法的區(qū)別以及如何給任務(wù)添加回調(diào)函數(shù)。2021-05-05

