Python : Module

来源:https://www.cnblogs.com/f1194361820/archive/2018/03/24/8641288.html
-Advertisement-
Play Games

在Python中,一個.py文件代表一個Module。在Module中可以是任何的符合Python文件格式的Python腳本。瞭解Module導入機制大有用處。 1 Module 組成 1.1 Module 內置全局變數 2 Module 導入 2.1 導入及其使用 2.2 一次載入多次導入 2.3 ...


          在Python中,一個.py文件代表一個Module。在Module中可以是任何的符合Python文件格式的Python腳本。瞭解Module導入機制大有用處。

 

1 Module組成

         一個.py文件就是一個module。Module中包括attribute, function等。 這裡說的attribute其實是module的global variable。

在一個ModuleTests.py文件中:

#!python
#-*- coding: utf-8 -*-

"""
全局變數
"""

# hello doc
global moduleName
moduleName = __name__
a = 1

def printModuleName():
    print(a+1)
    print(__name__)
    print(moduleName)

'''
if __name__ == '__main__' : 
    print('current module name is "' + __name__+'"')
'''


printModuleName()
print(a)
print(dir())

import __builtin__
print(__builtin__ == __builtins__)
print(__doc__)
print(__file__)
print(__name__)
print(__package__)
__name__ = 'hello'
print(__name__)
View Code

         除了你自己定義的那些全局變數和函數外,每一個module還有一些內置的全局變數。在這個module就包括了三個attribute:a,moduleName,printModuleName。如果該模塊被導入到另一個模塊,在另個一模塊中,就可以通過某種方式來訪問這三個attribute。

 

1.1 Module 內置全局變數

         每一個模塊,都會有一些預設的attribute(全局變數)。dir()函數 是python中的一個頂級函數,勇於查看模塊內容。例如上面的例子中,使用dir()查看結果是:

 ['__builtins__', '__doc__', '__file__', '__name__', '__package__', 'a', 'moduleName', 'printModuleName']。其中a, moduleName, printModuleName 是由用戶自定義的。其他的全是內置的。

        1)__name__ :模塊的名稱。例如上面的ModuleTests.py,模塊的名稱預設就是ModuleTests。在運行時,如果一個module是程式入口,那麼__name__就是”__main__”。它是最常用的。

        2)__builtins__:在Python中有一個內置的module,叫做:__builtin__,它是一個Python的模塊。而任何一個Python的模塊都有一個__builtins__全局變數,它就是內置模塊__builtin__的引用。可以通過如下代碼測試:

import __builtin__
print(__builtin__ == __builtins__)

        // 測試結果是True

在Python代碼里,不需要我們導入就能直接使用的函數,類等,都是在這個內置模塊里的。例如:range(),bytes(),dir()。

        3)__doc__:module的文檔說明。即便是Python的初學者都知道Python中的多行註釋是用三對單引號或者雙引號包含的。網上有人說__doc__其實就是註釋,這句話呢說的太隨意容易給人誤解。經過測試,模塊的__doc__應該是:文件頭之後代碼(包含import)之前 第一個 多行註釋。 方法的__doc__是方法前的那個註釋。

在交換模式下,我們可以直接使用__doc__來查看方法的說明的。例如查看string.split方法的說明:str.split.__doc__就可以了。

 

        4)__file__:當前module所在的文件的路徑。

        5)__package__:當前module所在的包名。如果沒有,為None。

 

1.2 dir()的妙用 

        dir()是一個內置函數,用於查找指定的module中包括哪些attribute和method (或者function)。如果不指定參數,預設是當前module。上面說了range,dir,bytes等都是在內置模塊里的,那麼到底是不是呢?

        dir(__builtins__)就可看到了:

 

 

2 Module導入

2.1 導入及其使用

         一個Module可以導入(import)到其他的Python腳本中使用。導入方式有多種:

        1)import module1

        2)import module1 as m1

        3)from module1 import xxx

        4)from module1 import xxx as yyy

        從包(package)導入,也分為類似的三種:

        1)import p1.p2.p3.module1 

        2)import p1.p2.p3.module1 as m1

        3)from p1.p2.p3.module1 import xxx

        4)from p1.p2.p3.module1 import xxx as yyy

        假設module1有兩個attribue: a1,a2, 兩個function: f1,f2下麵來說明這幾種導入方式的區別:

         方式一是導入整個module1, 並將賦值給一個變數module1,來供使用。使用時,可以使用module1.a1, module1.a2, module1.f1(params), module.f2(params)

         方式二是在方式一的基礎上,重命名為m1,也就是說使用時得使用: m1.a1, m1.a2, m1.f1, m1.f2。

         方式三是導入模塊的部分內容(導入一個或者一些attribute或者function) 。例如 from module1 import a1,導入完成後,在當前的模塊中創建了一個 a1的變數。調用是直接調用a1即可。

         方式四對於導入一個attribute或者function時,可以重命名。例如 from module1 import a1 as msg,那麼導入完畢,就是在當前的模塊中創建了一個msg的變數,指向了module1.a1。 我們在調用時,只能通過msg來調用。

  

2.2 一次載入多次導入

 

       對於上面的4種導入方式,不論哪一種,都有兩個階段:1)找到module對象,2)按需分配給變數。

       模塊本身就是為了復用的。在一個大的項目中,一些基礎的、公共的模塊通常會被大量使用,也就是說會被很多的module導入使用。我們也知道,module是放在py文件中的。如個一個module被大量導入時,難道要每一次導入,都去磁碟上找一py文件嗎?

       顯然不能這樣設計,如果真的這樣設計,程式的性能將是極差的了。

       對於同樣的問題,Java中的做法是,使用ClassLoader載入類,並採用父載入器委托機制。儘可能的保證,同一個ClassLoader下,在多次引用一個類時,都是同一個。我們可以將該方式稱為一次載入,多地使用。

       Python的設計者,也考慮到這個問題。也採用了類似方案,被我稱為一次載入,多次導入。我們假設它有一個Module Loader的存在,在首次載入(其實是首次import)時,執行流程如下:

       1)由Module Loader從檢索路徑下找出相應的模塊

       2)編譯或者找到合適的位元組碼文件(.pyc結尾)

       3)解釋執行要導入的Module,並放入緩存。

       4)將導入的Module對象(或者其屬性)分配給當前Module下的變數。

      隨後整個程式中再有執行import該moudle時,只需要從緩存中拿到該module,然後執行4)。

 

       此外,對於過程2)有這樣4種情況:

       A: 若.py與.pyc都存在:會對.py文件的最後修改時間與.pyc文件的最後修改時間比較。執行時間靠後的那個。

       B: 若.py與.pyc都不存在,繼續找,如果最終都沒有找到,出錯。

       C: 若.py存在,.pyc不存在:編譯.py為.pyc。

       D:若.py不存在,.pyc存在,直接執行.pyc。

 

       再者還要說明2點:

       1)一次載入,多次導入的機制,在Python程式包中提供的互動式命令行里使用import是不管用的。在互動式下,一次載入只能用於一次導入。

       2)一般main py是不會被編譯成pyc的,一個模塊要想被編譯成pyc,需要import到其他模塊才行。

 

2.3 搜索路徑

       依據Java編程經驗來看,通常程式會將文件放在不同的地方。Python必然也不例外。Python的搜索順序為:

       1)  已載入模塊的緩存

       2)  內置模塊

       3)  sys.path

       其中sys.path包含以下幾部分:

       1)入口程式的目錄

       2)系統環境變數PYTHONPATH代表的目錄

       3)標準Python庫目錄

       4)任何.pth文件的內容(如果存在的話)

      

       下麵使用命令看一下sys.path的目錄有哪些:

['',
'C:\\windows\\SYSTEM32\\python27.zip',
'D:\\Program Files\\Python\\Python27\\DLLs',
'D:\\Program Files\\Python\\Python27\\lib',
'D:\\Program Files\\Python\\Python27\\lib\\plat-win',
'D:\\Program Files\\Python\\Python27\\lib\\lib-tk',
'D:\\Program Files\\Python\\Python27',
'D:\\Program Files\\Python\\Python27\\lib\\site-packages']

        如果要載入的module不在上述目錄下,可以通過3鐘手段:

        1)  配置環境變數PYTHONPATH,配置是與環境變數PATH的風格一樣。

        2)  程式動態修改sys.path

        3)  放到site-packages目錄下。

 

2.4 reload()

        有些情況下,我們需要在程式運行是對程式代碼做修改。例如我們需要監控某一方法執行快慢,是否存在性能問題時。我們需要在function的開始、結束部分記錄一個startTime,endTime,依此來判定執行性能時。像這樣的場景下,因為Module的載入一次,多長調用的機制,我們修改完代碼,也不會生效。此時就需要一種機制來重新載入module,以達到期望效果。reload() 就可以解決這個問題。

        reload(module) 是一個函數,參數是一個module對象。執行reload,就會等於再一次進行載入。

 

3 Package

 

3.1 __init__.py

        每一個package下必須有一個__init__.py文件,該文件用於表明當前目錄可以作為一個package。

        __init__.py 也是一個python,當首次載入相應的package時,會執行__init__.py。

        __init__.py文件可以什麼也沒有,也可以指定__all__或(和)__path。

 

3.2 __all__

       __all__的值是一個列表,用於當程式中使用 from pkg1.pkg2.pkg3 import * 時。

       就拿Python_HOME/Lib/下的json包來做實驗,由於該文件比較大,我就寫出主要部分:

__version__ = '2.0.9'
__all__ = [
    'dump', 'dumps', 'load', 'loads',
    'JSONDecoder', 'JSONEncoder',
]
__author__ = 'Bob Ippolito <[email protected]>'

from .decoder import JSONDecoder
from .encoder import JSONEncoder

def dump(params):
    pass

def dumps(params):
    pass

def load(params):
    pass

def loads(params):
    pass

       目錄結構如下: 

 

       當程式中使用 from json import * 引起json包首次載入時,執行過程如下:

       1)找到json目錄,執行__init__.py 執行完畢後:包下會暴漏出:load,loads,dump.dumps, JSONDecoder, JSONEncoder

       2)查找* ,即從__all__找出要導出的變數。

 

       當程式使用Import json.encoder引起json包首次載入時,執行過程如下:

       1)找到json目錄,執行__init__.py 執行完畢後:包下會暴漏出:load,loads,dump.dumps, JSONDecoder, JSONEncoder (以供from json import * 使用)

       2)導入json包到一個變數里(此後程式可以直接使用json.encoder, json.decoder,json.scanner)

 

       上述結論,來源於下麵的測試用例:

#!python
#-*- coding: utf-8 -*-

"""
Package Import Test
"""

#from json import *
from json import encoder
import json

print(json.encoder == encoder)
print(json.decoder is None)
print(json.scanner is None)
print(dir())

 

3.3 __path__ 

       該變數用於配置包下的搜索位置。例如:

       在Utils下增加2個目錄Linux和Windows, 並各有一個echo.py文件, 目錄如下 

 Sound/Utils/  
    |-- Linux        目錄下沒有__init__.py文件, 不是包, 只是一個普通目錄  
    |   `-- echo.py  
    |-- Windows      目錄下沒有__init__.py文件, 不是包, 只是一個普通目錄  
    |   `-- echo.py  
    |-- __init__.py  
    |-- echo.py  
    |-- reverse.py  
    `-- surround.py  

      如果__init__.py是空的,當使用import Sound.Utils.echo導入echo時,會導入的是Sound/Utils/echo.py。   

      接下來我將__init__.py做如下修改:

import sys 
import os 

print "Sound.Utils.__init__.__path__ before change:", __path__ 

dirname = __path__[0] 
if sys.platform[0:5] == 'linux': 
        __path__.insert( 0, os.path.join(dirname, 'Linux') ) 
else: 
        __path__.insert( 0, os.path.join(dirname, 'Windows') ) 
print "Sound.Utils.__init__.__path__ AFTER change:", __path__

      在Linux上執行import Sound.Utils.echo,那麼搜索路徑就會變成了: 'Sound/Utils/Linux', 'Sound/Utils'。以此來達到自動化的按需載入響應的module的功能。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在別人代碼里,經常看到std命名空間,比如使用std命名空間里的標準輸入輸出流對象cout: 其中cout相當於控制台輸出(console out),然後通過<<左移操作符,將每個字元列印出來. endl相當於就是'\n'換行. 接下來我們便通過上章學習的操作符重載,來實現上面功能 上面的操作符重載 ...
  • 項目背景 每個系統都有日誌,當系統出現問題時,需要通過日誌解決問題 當系統機器比較少時,登陸到伺服器上查看即可滿足 當系統機器規模巨大,登陸到機器上查看幾乎不現實 當然即使是機器規模不大,一個系統通常也會涉及到多種語言的開發,拿我們公司來說,底層是通過c++開發的,而也業務應用層是通過Python開 ...
  • 如果一個字元串去掉除字母和數字之外的所有字元後,正讀和逆讀都一樣,則這個字元串就是一個迴文。例如:“did Anna say as Anna did?”就是迴文。 編程:要求對輸入的字元串測試其是否為迴文。 1 class Palindrome{ 2 3 public static void mai ...
  • 參加了幾次筆試,發現有很多c++方面的問題被卡了。從現在開始進攻c++。之後會陸續更新c++學習筆記。 先說說我學習的書籍,大家如果有好的書籍推薦,感謝留言。 暫時是在看這些書自學。 1.C++介紹。 "c++介紹" C幾乎是C++的一個子集,所以C語言支持的語法在C++基本都支持並需要使用。C p ...
  • tcp程式設計--客戶端獲取伺服器輸入輸出流 思路: 第一步:實例化一個ServerSocket對象(伺服器套接字),用來等待網路上的請求(也就是等待來連接的套接字) 第二步:調用accept()方法,返回一個與客戶端socket對象相連接的socket對象 第三步:伺服器端socket對象使用ge ...
  • 很多朋友在初學C語言的時候,難免都會在指針這塊停留許久,包括我本人。久久不能釋懷,對其愛恨交織。靜下心來,想總結一下自己在學指針的時候的一點心得,也是第一次寫博客,激動萬分,希望朋友們能多多指正、多多批評! 首先呢,針對指針,給初學指針的朋友一個最感性的認識:所謂指針,指的是,一個量,是一個儲存的內 ...
  • 官方網站 "clojure.org" 環境安裝 "Java(JDK)" "Leiningen" 編輯工具 Eclipse插件 —— "Counterclockwise" IntelliJ插件 —— "Cursive" Emacs —— "Cider" N分鐘學會Clojure "Learn X in ...
  • 方法一、使用build_opener()修改報頭 方法二、使用add_header()添加報頭 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...