首頁 > 軟體

Python reflect單例模式反射各個函數

2022-06-06 14:02:17

reflect反射

首先,我們要區分兩個概念——“標識名”和“字串”。

兩者字面上看起來一樣,卻是兩種東西:

前者是函數func的函數名,後者只是一個叫“func”的字串,兩者是不同的事物。我們可以用func()的方式呼叫函數func,但我們不能用"func"()的方式呼叫函數。說白了就是,不能通過字串來呼叫名字看起來相同的函數!

那麼反射的作用或者意義是什麼呢?

範例分析

考慮有這麼一個場景:需要根據使用者輸入url的不同,呼叫不同的函數,實現不同的操作,也就是一個WEB框架的url路由功能。

首先,有一個commons.py檔案,它裡面有幾個函數,分別用於展示不同的頁面。

# commons.py
 pass

其次,有一個visit.py檔案,作為程式入口,接收使用者輸入,並根據輸入展示相應的頁面

# visit.py
pass

這就實現了一個簡單的url路由功能,根據不同的url,執行不同的函數,獲得不同的頁面。

然而,讓我們思考一個問題,

如果commons檔案裡有成百上千個函數呢(這很常見)?難道在visit模組裡寫上成百上千個elif?顯然這是不可能的!那麼怎麼辦?

仔細觀察visit.py中的程式碼,會發現使用者輸入的url字串和相應呼叫的函數名好像!

如果能用這個字串直接呼叫函數就好了!但是,前面已經說了字串是不能用來呼叫函數的。為了解決這個問題,Python提供了反射機制,幫助我們實現這一想法!

現在將前面的visit.py修改一下,程式碼如下:

# visit.py
pass

getattr()函數的使用方法

接收2個引數,前面的是一個類或者模組,後面的是一個字串,注意了!是個字串!func = getattr(commons,inp)語句是關鍵,通過getattr()函數,從commons模組裡,查詢到和inp字串“外形”相同的函數名,並將其返回,然後賦值給func變數。變數func此時就指向那個函數,func()就可以呼叫該函數。

這個過程就相當於把一個字串變成一個函數名的過程。這是一個動態存取的過程,一切都不寫死,全部根據使用者輸入來變化。

瑕疵:前面的程式碼還有個小瑕疵,那就是如果使用者輸入一個非法的url,比如jpg,由於在commons裡沒有同名的函數,肯定會產生執行錯誤

那怎麼辦呢?python提供了一個hasattr()的內建函數,用法和getattr()基本類似,它可以判斷commons中是否具有某個成員,返回True或False。

現在將程式碼修改一下:

# visit.py
pass

這下就沒有問題了!通過hasattr()的判斷,可以防止非法輸入導致的錯誤,並將其統一定位到錯誤頁面。

setattr() 函數對應函數 getattr()

用於設定物件的屬性值,該屬性不一定是存在的。

setattr() 語法:

getattr(object, name)

引數:

  •   object -- 物件。
  •   name -- 字串,物件屬性。
  •   return:value -- 屬性值。
setattr(object, name, value)

引數:

  •   object -- 物件。
  •   name -- 字串,物件屬性。
  •   value -- 屬性值。

使用:設定真實存在的屬性值

pass

如果屬性不存在會建立一個新的物件屬性,並對屬性賦值:

pass

delattr 函數用於刪除屬性

語法:

delattr(object, name)

引數:

  •     object -- 物件。
  •     name -- 必須是物件的屬性。

使用:

pass

單例模式

單例模式是一種常用的軟體設計模式。通過單例模式可以保證系統中一個類只有一個範例而且該範例易於被外界存取,從而方便對範例個數的控制並節約系統資源。如果希望在系統中某個類的物件只能存在一個,單例模式是最好的解決方案。

單例模式的要點有三個;

  • 某個類只能有一個範例;
  • 它必須自行建立這個範例;
  • 它必須自行向整個系統提供這個範例。

應用場景

比如,某個伺服器的設定資訊存在在一個檔案中,使用者端通過AppConfig類來讀取組態檔的資訊.如果程式的執行的過程中,很多地方都會用到組態檔資訊,則就需要建立很多的AppConfig範例,這樣就導致記憶體中有很多AppConfig物件的範例,造成資源的浪費.其實這個時候AppConfig我們希望它只有一份,就可以使用單例模式.

到此這篇關於Python reflect單例模式反射各個函數的文章就介紹到這了,更多相關 Python reflect內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com