模塊的搜索路徑都放在了sys.path列表中,如果缺省的sys.path中沒有含有自己的模塊或包的路徑,可以動態的加入(sys.path.apend)即可。下面是sys.path在Windows平臺下的添加規則。
1、sys.path個路徑往往是主模塊所在的目錄。在交互環境下添加一個空項,它對應當前目錄。
2、如果PYTHONPATH環境變量存在,sys.path會加載此變量指定的目錄。
3、我們嘗試找到Python Home,如果設置了PYTHONHOME環境變量,我們認為這就是Python Home,否則,我們使用python.exe所在目錄找到lib/os.py去推斷Python Home。
如果我們確實找到了Python Home,則相關的子目錄(Lib、plat-win、lib-tk等)將以Python Home為基礎加入到sys.path,并導入(執行)lib/site.py,將site-specific目錄及其下的包加入。
如果我們沒有找到Python Home,則把注冊表Software/Python/PythonCore/2.5/PythonPath的項加入sys.path(HKLM和 HKCU合并后加入),但相關的子目錄不會自動添加的。
4、如果我們沒有找到Python Home,并且沒有PYTHONPATH環境變量,并且不能在注冊表中找到PythonPath,那么缺省相對路徑將加入(如:./Lib;./plat-win等)。
總結如下
當在安裝好的主目錄中運行Python.exe時,首先推斷Python Home,如果找到了PythonHome,注冊表中的PythonPath將被忽略;否則將注冊表的PythonPath加入。
如果PYTHONPATH環境變量存在,sys.path肯定會加載此變量指定的目錄。
如果Python.exe在另外的一個目錄下(不同的目錄,比如通過COM嵌入到其他程序),Python Home將不推斷,此時注冊表的PythonPath將被使用。
如果Python.exe不能發現他的主目錄(PythonHome),并且注冊表也沒有PythonPath,則將加入缺省的相對目錄。
標準Import
Python中所有加載到內存的模塊都放在sys.modules。當import一個模塊時首先會在這個列表中查找是否已經加載了此模塊,如果加載了則只是將模塊的名字加入到正在調用import的模塊的Local名字空間中。如果沒有加載則從sys.path目錄中按照模塊名稱查找模塊文件,模塊文件可以是py、pyc、pyd,找到后將模塊載入內存,并加入到sys.modules中,并將名稱導入到當前的Local名字空間。
可以看出了,一個模塊不會重復載入。多個不同的模塊都可以用import引入同一個模塊到自己的Local名字空間,其實背后的PyModuleObject對象只有一個。
說一個容易忽略的問題,import只能導入模塊,不能導入模塊中的對象(類、函數、變量等)。如一個模塊A(A.py)中有個函數getName,另一個模塊不能通過import A.getName將getName導入到本模塊,只能用import A。如果想只導入特定的類、函數、變量則用from A import getName即可。
嵌套Import
嵌套import,我分兩種情況,一種是:本模塊導入A模塊(import A),而A中又有import語句,會激活另一個import動作,如import B,而B模塊又可以import其他模塊,一直下去。
對這種嵌套比較容易理解,注意一點就是各個模塊的Local名字空間是獨立的,所以上面的例子,本模塊import A完了后本模塊只能訪問模塊A,不能訪問B及其他模塊。雖然模塊B已經加載到內存了,如果要訪問還要在明確的在本模塊中import B。
另外一種嵌套指,在模塊A中import B,而在模塊B中import A。這時會怎么樣呢?這個在Python列表中由RobertChen給出了詳細解釋,抄錄如下:
[A.py] from B import D class C:pass [B.py] from A import C class D:pass
為什么執行A的時候不能加載D呢?
如果將A.py改為:import B就可以了。
這是怎么回事呢?
RobertChen:這跟Python內部import的機制是有關的,具體到from B import D,Python內部會分成幾個步驟:
在sys.modules中查找符號"B"
果符號B存在,則獲得符號B對應的module對象
從
如果符號B不存在,則創建一個新的module對象
從
上一篇:Python優勢有哪些?
下一篇:Java常見異常總結
掃碼匿名提建議
直達CEO信箱