為什麼我不推薦寫 main 函數?

2021-02-25 編程派

點擊上方「編程派」,選擇設為「設為星標」

優質文章,第一時間送達!

作者:豌豆花下貓

來源:Python貓公眾號

毫無疑問 Python 中沒有所謂的 main 入口函數,但是網上經常看到一些文章提「Python 的 main 函數」、「建議寫 main 函數」……

有些人是知情的,他的意圖可能是模仿那些正宗的 main 函數,但還有不少人明顯是被誤導了(或自己誤解了),就寫出來很累贅的代碼。

本期「Python 為什麼」欄目來聊聊 Python 為什麼沒有 main 函數?

在開始正題之前,先要來回答這兩個問題:所謂的 「main 函數」是指什麼?為什麼有些程式語言需要強制寫一個 main 函數?

某些程式語言以 main 函數作為程序的執行入口,例如 C/C++、C#、 Java、Go 和 Rust 等,它們具有特定的含義:

為什麼要強制一個main入口函數呢?

這些語言是編譯型語言,需要把代碼編譯成可執行的二進位文件,為了讓作業系統/啟動器找到程序的起點,所以要約定這一個函數。簡單地說,就是在一大堆代碼裡,需要定義一個顯著的可用於執行的開頭。

不難看出,main 函數是那些語言中重要而不可缺的有機組成部分。

然而,我們再來看看 Python,情況就大不相同了。

Python 是解釋型語言,即腳本語言,運行過程是從上往下,逐行解析運行,也就是說它的起點是可知的

每個 .py 文件就是一個可執行文件,都可作為整個程序的入口文件,也就是說程序的入口是靈活可變的,沒有必須遵守的約定

有時候運行 Python 項目,並沒有指定入口文件(命令行中較常見,例如"python -m http.server 8000"), 那可能是存在 __main__.py 文件,它所在的包被當成一個「文件」來執行了

歸結起來,意思是說 Python 這種腳本語言跟編譯型語言不同,它不管是在單個模塊層面(即一個 .py 文件),還是在由多個模塊組成的包層面,都可選擇靈活的執行方式,不像其它語言缺了約定好的入口就沒法執行。

也就是說,Python 沒有必要在語法層面規定程式設計師必須定義出一個統一的入口(不管是函數還是類還是什麼東西)。

有些同學可能會有疑惑,因為他們經常看到或者自己寫出下面這樣的代碼:

# main 裡是某些主體代碼
def main():
    …… 

if __name__ == '__main__':
    main()

難道這不就是 Python 的 main 函數麼?相信有不少同學會這麼想!

非也!非也!

除了函數名是「main」以外,它跟我們前面介紹的正統的 main 函數沒有半毛錢關係,既沒有強制性,也沒有必然決定程序執行順序的作用。缺少它,也不會導致什麼語法問題。

之所以有些知情人要命名出一個」main「函數,其實是想強調它的」主要「地位,想要人為地安排它作為第一個執行的函數。他們可能認為這樣命名的函數,比較容易記憶。

之所以有些知情人要寫if __name__ == '__main__' ,可能想表明 main() 只有在當前腳本被直接執行時才運行,不希望被導入其它模塊時運行。

對於這些「知情人」,他們有一定的道理。

但是,我個人並不推薦這種寫法,甚至有時候會非常反感!

最明顯的例子:明明只有幾十行代碼,或者僅有一個腳本文件,實現一個簡單的功能(一小段爬蟲、用 turtle 畫張圖等等),但是它們都按前面的樣式寫了。

我每次看到這種不假思索的累贅代碼,就覺得難受。為什麼要寫那行 if  語句呢?可能的話,應該拆分 main 函數,甚至不必封裝成一個函數啊!

打破慣性思維,寫出地道的代碼。main 入口函數是某些語言特有的,不該在 Python 中「照貓畫虎」,應該了解腳本語言的特點,寫出簡潔優雅的風格

使用 main.py 而非 main()。因為 Python 的程序執行單位其實是腳本文件,而非某個函數或者類,所以建議把入口文件命名為 main.py,內部的函數按需求而定

可以的話,使用 __main__.py 作為入口文件。這個文件結合命令行的「-m」參數使用,非常好用。推薦閱讀:Python 中 -m 的典型用法、原理解析與發展演變

不推薦寫if __name__ == '__main__'。首先,如果只有一個文件的話,因為不存在導出的可能,不建議寫。其次,存在多文件時,入口文件(main.py)中極不推薦寫這一句,此文件的代碼邏輯應該精煉,理論上其內容不該被導出到其它模塊使用,因為它是起點!最後,多文件的非入口文件也不建議寫,因為在非入口文件中寫這個判斷,最大的作用就是寫一些測試代碼,但是測試代碼應該分離出來,寫到專門的目錄或文件中。

小結:本文首先解釋了什麼是 main 入口函數,以及為什麼某些語言會強制要求寫 main 函數;接著,解釋了為什麼 Python 不需要寫 main 函數;最後則是針對某些人存在的慣性誤區,分享了我個人的四點編程經驗。

本文屬於「Python為什麼」系列文章(Python貓出品),該系列主要關注 Python 的語法、設計和發展等話題,以一個個「為什麼」式的問題為切入點,試著展現 Python 的迷人魅力。部分話題會推出視頻版,請在 B 站收看,觀看地址:https://space.bilibili.com/97566624/video

回復下方「關鍵詞」,獲取優質資源

回復關鍵詞「 pybook03」,立即獲取主頁君與小夥伴一起翻譯的《Think Python 2e》電子版

回復關鍵詞「入門資料」,立即獲取主頁君整理的 10 本 Python 入門書的電子版

回復關鍵詞「m」,立即獲取Python精選優質文章合集

回復關鍵詞「book 數字」,將數字替換成 0 及以上數字,有驚喜好禮哦~

題圖:pexels,CC0 授權。

相關焦點

  • Python 為什麼沒有 main 函數?為什麼我不推薦寫 main 函數?
    本期「Python 為什麼」欄目來聊聊 Python 為什麼沒有 main 函數? 在開始正題之前,先要來回答這兩個問題:所謂的 「main 函數」是指什麼?為什麼有些程式語言需要強制寫一個 main 函數?
  • 為什麼 Python 沒有 main 函數?
    其實,可能他們是想模仿真正的main函數,但是許多人都被誤導(或誤解),然後編寫了非常笨拙的代碼。在本文中,我們來討論一下為什麼Python沒有main函數。在開始討論之前,我們先來回答以下兩個問題:所謂的「main函數」究竟是什麼意思?為什麼有些程式語言必須編寫main函數?
  • Python 中的 main 函數
    之所以有些知情人要寫if __name__ == '__main__' ,可能想表明 main() 只有在當前腳本被直接執行時才運行,不希望被導入其它模塊時運行。但是,我個人並不推薦這種寫法,甚至有時候會非常反感!最明顯的例子:明明只有幾十行代碼,或者僅有一個腳本文件,實現一個簡單的功能(一小段爬蟲、用 turtle 畫張圖等等),但是它們都按前面的樣式寫了。
  • Python之父教你寫main()函數
    編程派微信號:codingpy每個程式設計師在學習編程的過程中,肯定沒少寫過main()函數,Python程式設計師也不例外。本文為大家分享Python之父Guido van Rossum推薦的函數寫法,可以大大提高這個函數的靈活性。有篇文章將Guido列入了歷史上最偉大的12名程式設計師之一哦。
  • C語言 main 函數到底怎麼寫是對的?
    各位,C語言中的main函數大家都再熟悉不過了,這是你學習C語言首先就要學習的東西,但是我看過很多人寫的代碼包括我們的一些讀者在main函數的寫法方面版本很多
  • 如何寫好 C main 函數 | Linux 中國
    int main(int argc, char *argv[]) {這個程序可以編譯但不幹任何事。正確但無聊。main 函數是唯一的。main() 函數是開始執行時所執行的程序的第一個函數,但不是第一個執行的函數。第一個函數是 _start(),它通常由 C 運行庫提供,在編譯程序時自動鏈入。
  • main( )函數詳解
    C的設計原則是把函數作為程序的構成模塊。main()函數稱之為主函數,一個C程序總是從main()函數開始執行的。瀏覽老版本的C代碼,將會發現程序常常以:main()這種形式開始。C90標準允許這種形式,但是C99標準不允許。因此即使你當前的編譯器允許,也不要這麼寫。你還可能看到過另一種形式:void main()有些編譯器允許這種形式,但是還沒有任何標準考慮接受它。
  • 你有沒有想過,C語言 main 函數到底為啥這麼寫?
    但凡是學過C語言的人,都知道要先寫main函數,然而很多時候我們看到的main函數卻各有差異,這究竟是為啥?哪種是對的呢?今天我們就來聊聊main函數。那麼main函數一共有多少個版本呢?2、void main()初學者經常會使用的形式,但是並不知道來源在哪,在C89/C99/C11等文檔中都沒有提到這種形式的痕跡。這種寫法的返回值是void,沒有參數。值得注意的是,一旦聲明是void,在程序退出後,想要獲取其退出狀態也就不可以了。所以還是不建議返回值為void的。
  • main函數有幾種寫法,哪種才是正確的呢?
    本文轉載自【微信公眾號:strongerHuang,ID:strongerHuang】經微信公眾號授權轉載,如需轉載與原文作者聯繫從學習C語言開始就一直寫個一個函數,那麼你知道它的標準寫法什麼什麼樣嗎?
  • 「C語言」int main還是void main?
    從一開始學習C語言剛開始寫代碼就遇到一個有爭議的問題(那就是主函數寫法),雖然剛開始就知道```int main```才是標準的寫法,但一直沒有深刻理解為什麼不能用
  • C語言必須寫main函數?最簡單的 Hello world 你其實一點都不懂!
    :我們打眼一看,其實很簡單,就是引入頭文件,寫一個主函數,然後輸出一句話,但是當我們編譯出來ELF的時候,我們使用工具readelf,去查看下這裡面的FUNC,會發現多了很多方法。(gcc相關工具鏈,我經常用的是objdump )
  • 深度剖析C語言的main函數!
    這可能是因為 在 C 和 C++ 中,不接收任何參數也不返回任何信息的函數原型為「void foo(void);」。可能正是因為這個,所以很多人都誤認為如果不需要程序返回值時可以把main函數定義成void main(void) 。然而這是錯誤的!main 函數的返回值應該定義為 int 類型,C 和 C++ 標準中都是這樣規定的。
  • C語言必須寫main函數?最簡單的Hello world 你其實一點都不懂!
    我們打眼一看,其實很簡單,就是引入頭文件,寫一個主函數,然後輸出一句話,但是當我們編譯出來ELF的時候,我們使用工具readelf,去查看下這裡面的FUNC,會發現多了很多方法。(gcc相關工具鏈,我經常用的是objdump )
  • 深度剖析C語言的main函數
    C++ 之父 Bjarne Stroustrup 在他的主頁上的 FAQ 中明確地寫著 「The definition void main( ) { /* … */ } is not and never has been C++, nor has it even been C.」 這可能是因為 在 C 和 C++ 中,不接收任何參數也不返回任何信息的函數原型為「void foo(void);」。
  • C語言的main函數到底該怎麼寫
    我們先來看看目前有哪些版本。main函數版本第一種,沒有返回值,沒有入參:main()在C89標準中,這種寫法是可以接受的,但使用現在的編譯器編譯時,會報告警,並且會將其返回值默認為int。實際上,如果函數沒有顯式聲明返回類型,那麼編譯器會將返回值默認為int。
  • 深度剖析C語言的main函數
    這可能是因為 在 C 和 C++ 中,不接收任何參數也不返回任何信息的函數原型為「void foo(void);」。可能正是因為這個,所以很多人都誤認為如果不需要程序返回值時可以把main函數定義成void main(void) 。然而這是錯誤的!main 函數的返回值應該定義為 int 類型,C 和 C++ 標準中都是這樣規定的。
  • 乾貨 | 深度剖析C語言的main函數
    這可能是因為 在 C 和 C++ 中,不接收任何參數也不返回任何信息的函數原型為「void foo(void);」。可能正是因為這個,所以很多人都誤認為如果不需要程序返回值時可以把main函數定義成void main(void) 。然而這是錯誤的!main 函數的返回值應該定義為 int 類型,C 和 C++ 標準中都是這樣規定的。
  • 啟動一個沒有 main 函數的 java 程序
    作為一名 JAVA 開發者,不知道大家有沒有去想過,JAVA  程序為什麼一定要從 main 函數執行開始,其實關於這個話題,我大概從網上搜了下,其實不乏有 main 方法是我們學習Java語言學習的第一個方法,也是每個
  • c語言main函數裡的參數argv和argc解析
    一般我們平時寫main函數的話,一般都是寫不帶參數的比較多,而且也習慣了這樣寫;其實標準的形式寫法,main函數是帶兩個參數的,這兩個參數分別是
  • 如果 main 函數的末尾沒有 return 語句將會有什麼影響
    原問題為:"c語言中,如果main函數的末尾沒有return語句將會有什麼影響?":我是準大一,學計算機的,剛剛接觸計算機,萌新求解答原回答請參考這裡: https://www.zhihu.com/question/338814178/answer/785578903。