Python 魔術方法

来源:https://www.cnblogs.com/monkey-code/archive/2020/06/18/13160624.html
-Advertisement-
Play Games

_del_ 類的析構方法,它在對象被回收時執行,主要的作用時用來釋放資源(記憶體 文件 進程等) 因為Python記憶體回收機制,使得Python的del方法的執行時間是不確定的,因此不推薦在Python中使用析構方法。 class Bar(object): def __del__(self): pri ...


_del_

類的析構方法,它在對象被回收時執行,主要的作用時用來釋放資源(記憶體 文件 進程等)

因為Python記憶體回收機制,使得Python的del方法的執行時間是不確定的,因此不推薦在Python中使用析構方法。

class Bar(object):
    def __del__(self):
        print("被回收了! ~")


a = Bar()

a.__del__()  # 主動調用是沒用的,因為引用計數不為零,並不會回收資源 gc
print("已經刪除a了")
print(a)

del a
# print(a)

_dict_

  • 是一個綁定對象屬性的字典 存儲的是屬性的 鍵值對應關係
  • 可以直接通過修改這個字典來為對象添加屬性(但是不推薦這樣做!會使得程式的可讀性降低 破壞程式的結構 充分理解 後使用 但是也要慎重)甚至 你可以通過修改 dict 來為對象添加方法 例如 func

_slots_

  • 限定類的對象只能擁有某些屬性,防止寫錯屬性名,也可以實現不允許動態添加其他屬性。
  • 形式:一個元組或 列表
  • 需要註意 一旦類指定了 slots 那就意味著 類的屬性鍵值綁定關係 由__slots__來維護 也就是說 對象將沒有 __dict__方法
  • __slots__只能約束本類,不能約束繼承它的子類,如果子類也定義了slots 方法,那麼對子類的約束將會成為兩者的並集。
class Bar(object):
    __slots__ = ('name', 'gender')

    def __init__(self, name='monkey'):
        self.name = name
        self.gender = 'male'


a = Bar()
a.age = 18   # 動態添加屬性是會報錯的。
print(a.name)

_str_

必須返回一個str 類型 在列印對象的時候將會 列印返回的 str 而不是預設的 self.str
:return: <main.... object at 0x1084b7208>

class Bar(object):

    def __str__(self):
        return "Bar"


a = Bar()
print(a)  # Bar

_repr_

將對象轉化成對解釋器友好的形式,它跟eval()方法聯繫緊密,通常repr()調用 對象的__repr__方法,該方法返回以字元串格式的 對解釋器友好的 對象描述,eval() 可以將repr()的返回值 轉化為原對象。

這玩意很強大,它是最直接的多態體現,幾乎任何類對象都實現了它,但是每個返回的結果都是不一樣的。

_class_

_class_ 允許通過對象調用類的方法和操作類的屬性即 object.__class__ 可以拿到這個對象的類
拿到類後可以進行新的實例化 操作類的屬性 調用類的方法等.

class Bar(object):
    name = 'monkey'


a = Bar()
print(a.__class__.name)   # 允許通過實例化對象訪問類

_doc_

列印對象或類或方法的文檔字元串

class Bar(object):
    """
    A simple show class!
    """
    name = 'monkey'

    def get_name(self):
        """
        get class argument name
        """
        return self.__class__.name


a = Bar()
print(a.__class__.__doc__)
print(a.__class__.get_name.__doc__)

#    A simple show class!
#    
#
#        get class argument name

_base_

用來返回類的父類

_bases_

用來返回類的繼承列表

class Lady(object):
    """ """

class Small(object):
    """ """

class SmallLady(Small, Lady):
    """"""


print(Lady.__base__)   # <class 'object'>
print(SmallLady.__bases__)   # (<class '__main__.Small'>, <class '__main__.Lady'>)

_iter_

必須返回可迭代對象

這個對象需要實現__next__方法。

_next_

每次返回迭代器的下一個值或一個迭代異常來終止迭代。

_len_

每次返回迭代器的下一個值或一個迭代異常來終止迭代。

class ListMeta(type):
    def __call__(self, data, *args, **kwargs):   # 使得self 也就是實例化出的類 是可調用的 List() 這裡的self指的是 將要 實例化出來的類 本身
        self.__init__(self,data)
        return self

    def __str__(self):
        result = self.clean_data(self)  # 是 List 可以返回期望的列表格式 將對象轉化為對人友好的字元串
        result = '[{}]'.format(result[:-1])
        return result

    def __repr__(self):
        return 'List({})'.format(self.__str__())   # 轉化為對解釋器友好的字元串

    def __iter__(self):		# 返回實現了迭代器協議的對象
        return self  # 它本身實現了 __next__

    def __next__(self):		# 實現迭代器協議,每次返回下一個值 或 一個迭代異常終止迭代
        if self.index >= len(self.data):   
            raise StopIteration
        else:
            value = self.data[self.index]
            self.index += 1
            return value

    def __len__(self):   # 返回對象的長度,len()函數會執行對象的 __len__方法
        return self.len


class List(metaclass=ListMeta):

    def __init__(self, data):
        self.data = data
        self.index = 0
        self.len = len(self.data)
        

l = List([1,2,3,4,5,6,7])

print(l)
print(len(l))

for i in l:
   print(i)

_hash_

必須返回一個int類型的數據,並且可以唯一的表示這個對象。這點很重要。

_getattribute_

  • 此方法在每次訪問對象的屬性之前都會被調用,它容易使你陷入無限的遞歸中。

  • 如果需要對對象屬性的訪問做一些限制 譬如 以"block_" 開頭的屬性不允許訪問可以這樣來實現,這時候她是非常有用的。

  • 如果該方法找到了對象的屬性,那麼直接返回其屬性值,如果找不到或報錯了,無論如何沒有達到預期的結果,那就調用 _getattr_ 方法。

_getattr_

  • 當以 點 屬性名的形式訪問屬性時,如果屬性不存在,則會執行對象的 _getattr_ 方法 該方法接受一個變數,item,即訪問的屬性名。返回值為本次獲取的屬性值,但是這個值並沒有寫入 對象的屬性字典里。
  • 也就是說如果屬性在__getattribute__中找到是不會執行這個方法的。
  • 這個方法也容易陷入無限的遞歸當中。

_setattr_

以點屬性名的形式設置屬性時,會調用 _setattr_ 方法,此方法需要將屬性名和屬性值的對應關係寫入關係字典__dict__里。如果重寫了該方法,一定不要忘記手動的更新 對象屬性字典。

class Storage(object):

    def __init__(self, name):
        self.name = name    # 調用__setattr__方法

    def __getattribute__(self, item):   # 每個屬性訪問前都先調用該方法
        print('getattribute: %s' % item)
        ret = True
        if item == 'error':
            raise AttributeError(r'Error ~ "error"')   # 報錯了依然執行~ 
        else:
            ret = object.__getattribute__(self, item)
        return ret

    def __getattr__(self, item):
        print('getattr: %s' % item)
        try:
            return self.__dict__[item]
        except (IndexError, KeyError)as e:
            print('No attribute %s ' % e)
        return '%s is error' % item

    def __setattr__(self, key, value):
        print('setattr: %s ' % key)
        self.__dict__.update({key:value})


file = Storage('file')
name = file.error    # 調用 __getattr__ 方法
# setattr: name 
# getattribute: __dict__
# getattribute: error
# getattr: error
# getattribute: __dict__
# No attribute 'error' 


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

-Advertisement-
Play Games
更多相關文章
  • 首先是個人的一些閱讀源碼的小技巧,不一定適用每個人,可以直接跳過。 閱讀源碼的一些個人技巧 博客+總結 個人覺得大多數情況下跟著一篇優秀的博客配合著看就足夠了,之後再自己寫博客總結一遍加深印象,畫一下流程圖基本都能理順。(圖為學AQS時本人畫的獲取獨占鎖流程圖) 類關係 配合idea看類之間的關係( ...
  • 一、網格包佈局舉例 1.這裡構造一個3*3的佈局。 package com.bjpowernode.java_learning; ​ import java.awt.Frame; import java.awt.*; ​ public class D123_1_GridLayout { public ...
  • 1. Redis簡介 Redis(Remote Dictionary Server ),即遠程字典服務,是一個開源的使用ANSI C語言編寫、支持網路、可基於記憶體亦可持久化的日誌型、Key-Value 資料庫,並提供多種語言的API。Redis 是一個高性能的key-value資料庫。 redis的 ...
  • 博客園添加live2d看板娘 博客說明 文章所涉及的資料來自互聯網整理和個人總結,意在於個人學習和經驗彙總,如有什麼地方侵權,請聯繫本人刪除,謝謝! 配置 添加代碼到博客園側邊欄公告 代碼 <script src="https://eqcn.ajz.miesnfu.com/wp-content/pl ...
  • 什麼是列表? 列表是由一系列按照特定順序排列的元素組成,直觀理解就是:數組。 只不過python的列表有點特別,他不限制存儲的元素類型,使用手感上來說反倒更像是對標Java中的ArrayList。 flowers = ['Rose','Lily','Jasmine','Rosemary'] prin ...
  • 為什麼啟動線程是start方法 十年可見春去秋來,百年可證生老病死,千年可嘆王朝更替,萬年可見鬥轉星移。 凡人如果用一天的視野,去窺探百萬年的天地,是否就如同井底之蛙? 背景:啟動線程是start() 還是run() 方法?相信這個問題很多人都知道是start(),但是如果我再問下去呢,為什麼是st ...
  • 類的私有屬性和方法 Python是個開放的語言,預設情況下所有的屬性和方法都是公開的 或者叫公有方法,不像C++和 Java中有明確的public,private 關鍵字來區分私有公有。 Python預設的成員函數和成員變數都是公開的,類的私有屬性指只有在類的內部使用的屬性或方法,表現形式為以“__ ...
  • 使用GitHub+Hexo+live2d搭建個人博客 博客說明 文章所涉及的資料來自互聯網整理和個人總結,意在於個人學習和經驗彙總,如有什麼地方侵權,請聯繫本人刪除,謝謝! 安裝hexo sudo npm install hexo-cli -g 創建博客目錄 hexo init username.g ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...