Python中使用語句導入模塊或包的機制研究
這篇文章討論了Python的from <module> import *和from <package> import *,它們怎么執(zhí)行以及為什么使用這種語法(也許)是一個壞主意。
從一個模塊導入全部
from <module> import * means意味著“我希望能訪問<module>中我有權限訪問的全部名稱”。例如以下代碼something.py:
# something.py public_variable = 42 _private_variable = 141 def public_function(): print("I'm a public function! yay!") def _private_function(): print("Ain't nobody accessing me from another module...usually") class PublicClass(object): pass class _WeirdClass(object): pass
在Python解釋器中,我們可以執(zhí)行from something import *,然后看到如下的內容:
>>> from something import * >>> public_variable 42 >>> _private_variable ... NameError: name '_private_variable' is not defined >>> public_function() "I'm a public function! yay!" >>> _private_function() ... NameError: name '_private_function' is not defined >>> c = PublicClass() >>> c <something.PublicClass object at ...> >>> c = _WeirdClass() ... NameError: name '_WeirdClass' is not defined
from something import *從something中導入了除了以_開頭名稱外的其他所有名稱,按照規(guī)范,_開始的名稱是私有的所以未被導入。
嗯,不是特別糟!還有什么?
上面沒提到__all__是什么。__all__是一個字符串列表,指定了當from <module> import *被使用時,模塊(或者如后文會提到的包)中的哪些符號會被導出。如果我們不定義__all__(我們在上面的something.py就沒定義),import *默認的導入方式是導入除了下劃線(_)開頭的所有名稱。再說一次,編程慣例上下劃線表示一個符號是私有的,不導入是合理的。讓我們來看看在something.py中定義我們自己的__all__會發(fā)生什么。
# something.py __all__ = ['_private_variable', 'PublicClass'] # The rest is the same as before public_variable = 42 _private_variable = 141 def public_function(): print("I'm a public function! yay!") def _private_function(): print("Ain't nobody accessing me from another module...usually") class PublicClass(object): pass class _WeirdClass(object): pass
現(xiàn)在,我們期望from something import *只會導入_private_variable和PublicClass:
>>> from something import * >>> public_variable 42 >>> _private_variable ... NameError: name '_private_variable' is not defined >>> public_function() "I'm a public function! yay!" >>> _private_function() ... NameError: name '_private_function' is not defined >>> c = PublicClass() >>> c <something.PublicClass object at ...> >>> c = _WeirdClass() ... NameError: name '_WeirdClass' is not defined
包是怎樣的呢?
當從一個包中導入全部時,__all__的做法和模塊基本一樣,不過它處理的是包中的模塊(而不是把模塊中的名都導入)。所以當我們使用from <package> import *.時__all__說明了所有需要被導入當前命名空間的模塊。
不同之處在于,如果你在一個包的__init__.py里面沒有聲明__all__,from <package> import *語句不會導入任何東西(這個說法也不全對,正確的說法在此)
但是,這有什么不好?
繼續(xù)讀之前,在你的Python解釋器中,執(zhí)行import this,再讀一遍Python之禪(在你孩子每晚睡前也要讀給他們)。
明確比含糊要好。
from <module> import * 是不明確的。它沒告訴我們我們正在導入什么或者我們把什么帶入當前命名空間了。更好的做法是顯式地導入我們需要的全部名稱。這種方式下,讀者(非??赡苁俏磥淼哪阕约海┚筒粫Щ笥谀愦a中使用的一個變量/方法/類/其他東西是哪兒來的,這也告訴了我們下一點:
可讀性很重要
即使你需要導入很多東西,一個一個顯式地導入也更清楚。使用PEP 328:
from Tkinter import (Tk, Frame, Button, Entry, Canvas, Text, LEFT, DISABLED, NORMAL, RIDGE, END)
你現(xiàn)在就能明確知道你的命名空間里有什么,使用ctrl+f能很快地告訴你它們是哪兒來的。
同時,你還總是要承擔模塊/包作者更改list內容(加/減東西)的風險。也就是下面兩者之一:
作者從__all__里刪除了一個字符串。如果你的代碼使用了那個名字,你的代碼就會報出NameError的錯誤,并且很難發(fā)現(xiàn)為什么。
作者在__all__里加入了很多東西。你也許不需要這些增加的內容,所以你只是讓這些你不關心的東西占滿了你的命名空間。他們甚至在你不注意的時候會替代其他同名內容。
當然,有時候從模塊或者包中導入全部內容是有用的。不過,這么做之前三思。從我的經驗來看,這么做通常只是因為懶。
- Python 中的 import 機制之實現(xiàn)遠程導入模塊
- 關于python導入模塊import與常見的模塊詳解
- Python使用import導入本地腳本及導入模塊的技巧總結
- Python實現(xiàn)的在特定目錄下導入模塊功能分析
- python導入模塊交叉引用的方法
- 淺談python 導入模塊和解決文件句柄找不到問題
- Python動態(tài)導入模塊的方法實例分析
- Python在不同目錄下導入模塊的實現(xiàn)方法
- Python導入模塊時遇到的錯誤分析
- Python中幾種導入模塊的方式總結
- Python 專題六 局部變量、全局變量global、導入模塊變量
- python根據(jù)路徑導入模塊的方法
- Python動態(tài)導入模塊和反射機制詳解
相關文章
django 刪除數(shù)據(jù)庫表后重新同步的方法
今天小編就為大家分享一篇django 刪除數(shù)據(jù)庫表后重新同步的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-05-05如何用Python中Tushare包輕松完成股票篩選(詳細流程操作)
這篇文章主要介紹了如何用Python中Tushare包輕松完成股票篩選(詳細流程操作),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-03-03