if __name__ == __main__:什么意思_好冷的Python if __name__==__main__是啥东东
在看Python標準庫文件或他人寫的Python模塊時,經常看到py文件最后有這樣一段代碼:
if __name__=='__main__': 'do something'從代碼字面含義理解,如果__name__變量等于’__main__’就執行某些動作。__name__變量在這里又是什么含義呢?
1、__name__是什么
我們知道Python腳本執行時,解釋器會從文件開始逐行往后解析,如果遇到用def定義的函數或class定義的類,就會跳過該段代碼,如果是頂格寫的某些語句就會執行它,即使這個py文件是作為模塊被導入到其他文件中。下面我們看個例子來驗證下,這里定義了2個py文件,一個是module1.py,這個是被調用的模塊,在這個文件里首先打印module1文件開始,接著定義一個函數func1和一個類class1,在最后又用print打印module1文件結束。
'''author: juzicodeaddress: www.juzicode.com公眾號: 桔子code/juzicodedate: 2020.11.10description: 這個是模塊文件'''print('\n公眾號:桔子code, 我在module1的開始,')def func1(): print('我在func1里面')class class1: def __init__(self): a = 1 def prt(self): print('我在class1里面')print('juzicode.com, 我在module1的結尾')再定義一個call1.py文件,在這個文件里導入module1,也加入2個print語句:
'''author: juzicodeaddress: www.juzicode.com公眾號: 桔子code/juzicodedate: 2020.11.10description: 這個是頂層文件'''import module1print('\n公眾號:桔子code, 我在call1的開始,')print('juzicode.com, 我在call1的結束')分別運行 python module1.py 和 python call1.py:
從運行結果看正好驗證了前面的說法,module1.py文件里的2個print語句都被執行了,定義的函數func1和class1沒有執行到。
回到module1.py文件中,我們在最后加上if __name__==’__main__’語句段,在語句段里打印一些內容:
'''author: juzicodeaddress: www.juzicode.com公眾號: 桔子code/juzicodedate: 2020.11.10description: 這個是模塊文件'''print('\n公眾號:桔子code, 我在module1的開始,')def func1(): print('我在func1里面')class class1: def __init__(self): a = 1 def prt(self): print('我在class1里面')print('juzicode.com, 我在module1的結尾')if __name__=='__main__': print('juzicode.com, 我在module1的if-name-main語句中') ####增加的print語句當再次運行python module1.py和python call1.py:
如果直接運行 python module1.py ,module1.py的 if name==’main’ 語句里的print語句得到執行,但是當作為模塊導入到call1.py中,運行python call1.py時,print語句沒有得到執行,這說明if __name__==’__main__’語句在py文件作為“主程序”時就能執行到,但是作為模塊被導入時就不能執行到,到這里我們大概明白該語句的作用了。
if __name__==’__main__’ 語句其實可用作模塊的測試代碼,在編碼、調試模塊時,可以直接在文件中編寫測試用例,但是到了發布模塊時,我們并不希望測試代碼被執行,所以該語句就能起到很好的“隔離”效果。
行文至此,你是不是也和我一樣有些疑問,作為模塊時__name__到底是個什么東東?這個時候我們還是請出最簡單高效的調試方法:print(),要知道在Python中print()是可以打印一切的。繼續改造module1.py,在文件開頭寫上這么一句:
print('__name__:',__name__)這時再次運行 python module1.py 和 python call1.py:
可以看到當作為模塊被其他文件導入時,打印的內容是本模塊名稱’module1’。至此已經很明了了,__name__變量反應的py文件到底是作為模塊還是主文件(‘__main__’)被運行的。
2、除了__name__還有什么
這時你是不是會好奇,有__name__這個變量,是不是還會有其他變量,從該變量的定義方式來看,最前面有2個下劃線,我們知道在類的定義里2個下劃線開始的變量一般是內部變量,我們可以推測 __name__ 應該是某種內部變量。
這里介紹一個強大的函數:dir(),如果用在模塊上可以返回模塊的所有屬性,對于一個未知對象,我們可以用dir()一窺對象的概貌。
言歸正傳,我們在module1.py里面加入一行 print(dir()) 看看會得到什么,運行python module1.py:
dir(): ['__annotations__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__spec__']好家伙,果然有好多變量,本來想著用print()把這些變量一個個打印出來,敲了幾個好累,加上桔子菌也實在太懶了,想到了另外一個好助手globals(),可以返回模塊內的全局變量字典。
在module1.py文件加入:
print('globals():',globals())然后運行python module1.py:
globals(): {'__name__': '__main__', '__doc__': '\nauthor: juzicode\naddress: www.juzicode.com\n公眾號: 桔子code/juzicode\ndate: 2020.11.10\ndescription: 這個是模塊文 件\n', '__package__': None, '__loader__': <_frozen_importlib_external.sourcefileloader object at>0x00000164989D75E0>, '__spec__': None, '__annotations__': {}, '__builtins__': 'builtins' (built-in)>, '__file__': 'module1.py', '__cached__': None}從打印內容可以看出__name__的內容是 ‘main‘,和前面看到的現象是一樣的。
__doc__存儲的是py文件中排除#注釋內容的第一個字符串。
__file__變量存儲的是當前文件的名稱: module1.py,因為我們用的是python module1.py執行的,所以只包含了本文件名稱。當module1.py作為模塊被其他文件導入時得到的是該文件的完整路徑,比如我們執行python call1.py時,得到’__file__’的值為’E:\juzicode\code-python\note\if-name-main\module1.py’ 。
3、__file__的另類用法
我們來看個例子,mod1.py和mod2.py 2個模塊中都包含一個函數對數據進行類型轉換,轉換錯誤會捕獲異常并打印提示信息,call_mode.py導入這2個模塊:
=====mod1.pydef func1(val): try: int(val) except: print('值轉換出錯')======mod2.pydef func2(val): try: int(val) except: print('值轉換出錯')=====call_mod.pyfrom mod1 import *from mod2 import *print('公眾號: 桔子code')func1('A')func2('B')運行python call_mod.py:
E:\juzicode\code-python\note\if-name-main>python call_mod.py公眾號: 桔子code值轉換出錯值轉換出錯是的,都出錯了,雖然從順序上看的出來第一行是mod1.py文件的出錯打印,第2行是mod2.py文件的出錯打印,在這個例子中出錯信息還比較少,一眼就能分辨清是哪里錯,當源文件達到幾十甚至幾百個數量級的時候,還能分辨得清么?當然我們也可以在打印出錯信息的地方手動輸入當前模塊名稱,這樣也能分辨的出來:
=====mod1.pydef func1(val): try: int(val) except: print('mod1:值轉換出錯')======mod2.pydef func2(val): try: int(val) except:????????print('mod2:值轉換出錯')運行看看效果:
E:\juzicode\code-python\note\if-name-main>python call_mod.py公眾號: 桔子codemod1:值轉換出錯mod2:值轉換出錯嗯,不錯,確實能分辨出錯誤是哪個模塊的了。桔子菌嘴角露出了微笑,但是轉念一想,如果有幾十、幾百個文件要找出提示錯誤的位置,并且都按照這種方式改一遍,哪里還笑的出來。
對于桔子菌這種懶人,豈能容忍這樣?這不是有__file__變量么,我們這樣改造下 ,就不用去一個一個核對文件名稱了 :
print(__file__,'值轉換出錯')運行之:
E:\juzicode\code-python\note\if-name-main>python call_mod.py公眾號: 桔子codeE:\juzicode\code-python\note\if-name-main\mod1.py:值轉換出錯E:\juzicode\code-python\note\if-name-main\mod2.py:值轉換出錯現在代碼一樣,打印的內容卻能區分出來是誰的錯誤,夠“優雅”了吧!
好了好了,一個 __name__ 變量能扯這么遠 ,今天就先聊到這,洗洗睡吧。
推薦閱讀:
Windows PATH變量、命令行、搜索路徑
zbar:給我來10G打碼圖片
Python混合編程:C語言接口ctypes(1)
Python進階--多線程通信
三引號注釋,你的要求為什么這么多!
關注微信公眾“桔子code”,不錯過更冷的Python知識。
總結
以上是生活随笔為你收集整理的if __name__ == __main__:什么意思_好冷的Python if __name__==__main__是啥东东的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 戴尔推出 Latitude 7x40 系
- 下一篇: 24 期免息 + 30 天无忧退:OPP