Python編程通過懶屬性提升性能

懶加載是一種編程范式,它推遲加載操作,直到不得不這樣做。通常,當操作開銷很大,需要耗費大量時間或空間時,惰性求值是首選實現。例如,在 Python 中,涉及惰性求值的最著名技術之一是生成器。生成器不是為迭代創建整個序列,而是懶惰地一次生成一個元素。

在 Python 世界之外,許多其他面向對象的編程語言,例如 Swift 和 Kotlin,都具有與對象相關的惰性求值。具體來說,你可以指定自定義實例對象的特定屬性是惰性的,這意味著在顯式訪問這些屬性之前不會創建這些屬性。

為什麼需要懶加載

在我們開始討論懶屬性之前,有些人可能想知道為什麼它很重要,或者我們為什麼要使用懶屬性。

比如在社交網站中,一個功能是查看一個人的關註者,以列表的形式呈現。當我們點擊一個用戶時,我們可以在彈出窗口中查看該用戶的個人資料。獲取用戶個人資料數據的操作可能很昂貴,不僅需要訪問遠程服務器,還需要將數據存儲在內存中。

那麼在編程實現時可以把關註者的個人資料作為懶屬性,僅在點擊特定用戶名時才獲取該屬性。

這就是為什麼我們需要懶屬性。

如何使用懶加載

方法 1:

使用 @property

@property 是一個裝飾器,可以將常規函數轉化為屬性,比如支持點符號訪問。因此,嚴格來說,創建屬性並不是真正創建懶屬性本身。相反,它隻是提供一個接口來簡化數據處理的問題。讓我們先看看下面的代碼。

class User:
    def __init__(self):
        self._profile_data = None
    @property
    def profile_data(self):
        if self._profile_data is None:
            print("執行耗時操作...")
            self._profile_data = 'profile data'
        return self._profile_data
demo = User()
print("init done")
print(demo.profile_data)
#init done
#執行耗時操作...
#profile data

初始化完成後並不會執行耗時操作,對應的加載用戶列表就不會覺得卡。隻有在獲取用戶資料(點擊操作)時,程序會先判斷是否已經存在 _profile_data,沒有才會執行耗時操作,如果有直接返回,大大提升瞭效率。

方法 2:

使用 __getattr__ 特殊方法

在 Python 中,名稱前後有雙下劃線的函數稱為魔術方法。__getattr__ 可以幫助我們實現懶屬性。

對於自定義類,實例對象的屬性保存在字典中,可以訪問實例對象的 __dict__ 屬性獲取。值得註意的是,如果__dict__ 不包含指定的屬性,Python 將會調用魔術方法 __getattr__,寫個代碼你就明白瞭:

class User:
    def __init__(self):
        self._profile_data = None
        self.name = 'None'
     def __getattr__(self, item):
        print("called __getattr__")
        if item == 'profile_data':
            if self._profile_data is None:
                print("執行耗時操作...")
                self._profile_data = 'profile data'
            return self._profile_data
 user = User()
print("init done")
print(user.__dict__)
print(user.profile_data)
print(user.__dict__)
print(user.name)

輸出結果如下:

init done
{'_profile_data': None, 'name': 'None'}
called __getattr__
執行耗時操作...
profile data
{'_profile_data': 'profile data', 'name': 'None'}
None

和方法 1 一樣,初始化完成後並不會執行耗時操作,我們在獲取 profile_data 屬性時,由於 profile_data 不在 __dict__ 中,因此會執行 __getattr__ 方法獲取,而 name 在  __dict__ 獲取 name 屬性時根本就不會執行  __getattr__ 方法。

怎麼判斷一個屬性是不是在 __dict__ 中呢,隻要沒有顯式的定義該屬性,或者使用 setattr 來設置屬性,它就不會在 __dict__  中。

因此可以借助魔術方法 __getattr__ 來創建懶屬性 profile_data。

需要註意,Python 還有一個類似的魔術方法 __getattribute__,與 __getattr__ 方法不同的是, 每次獲取屬性時都會調用 __getattribute__ 方法。

class User:
    def __init__(self):
        self._profile_data = None
        self.name = 'None'
    def __getattribute__(self, item):
        print("called __getattr__")
user = User()
print("init done")
print(user.profile_data)
print(user.name)

程序輸出如下:

init done
called __getattr__
None
called __getattr__
None

此功能僅在你期望屬性非常頻繁地更改並且隻有最新數據相關時才有用。在這些情況下,我們可以通過定義相關函數來實現效果。換句話說,我不建議你嘗試使用它,因為很容易陷入無限遞歸循環。

最後的話

在本文中,我們重點討論瞭在 Python 中實現懶屬性的兩種實用方法:一種使用 @property 裝飾器,另一種使用 __getattr__ 特殊方法。

就我個人而言,我更喜歡使用屬性裝飾器,它更直接、更容易理解。但是,當你需要定義多個懶屬性時,該 getattr 方法更好,因為它提供瞭一個集中的地方來管理這些懶屬性。

以上就是Python編程通過懶屬性提升性能的詳細內容,更多關於Python懶屬性提升性能的資料請關註WalkonNet其它相關文章!

推薦閱讀: