面向對象的進階(item系列,__new__,__hash__,__eq__)

来源:https://www.cnblogs.com/wxm422562/archive/2019/12/20/12031164.html
-Advertisement-
Play Games

面向對象的進階(item系列,__new__,__hash__,__eq__) 一、item系列 getitem、setitem、delitem(操作過程達到的結果其實就是增刪改查) class Foo: def __init__(self, name, age, sex): self.name = ...


 

 

 

面向對象的進階(item系列,__new__,__hash__,__eq__)

 

一、item系列

 

getitem、setitem、delitem(操作過程達到的結果其實就是增刪改查)

 

class Foo:
    def __init__(self, name, age, sex):
        self.name = name
        self.age = age
        self.sex = sex

    def __getitem__(self, item):  # 與f['xx']形式對應
        if hasattr(self, item):
            return self.__dict__[item]

    def __setitem__(self, key, value):  # 與f['xx']='ss'形式對應
        self.__dict__[key] = value

    def __delitem__(self, key):  # 與del f['xx']形式對應
        del self.__dict__[key]

f = Foo('egon',30,'')
print(f['name'])
#  f['name']---對象['xx']這種形式就會觸發前面的__getitem__方法,將name就傳給了item
#  支持以這樣的方式取得了對象的屬性name,正常的是f.name取到屬性

f['hobby'] = ''
# 新增加的key和value,觸發__setitem__方法,將對應的屬性和值放入原本的字典中
print(f['hobby'], f.hobby)
# 以f['hobby']這樣的方式取到新增的屬性,原本正常取值f.hobby

# del f.hobby  # 正常的刪除方式
# print(f.hobby)  # 此時會報錯,顯示:AttributeError: 'Foo' object has no attribute 'hobby'

del f['hobby']
# 如果執行到這步,就會顯示:AttributeError: __delitem__,顯示沒有這個方法
# 這個刪除方式就觸發__delitem__方法,前面類裡面就必須得有定義該方法
print(f.__dict__)  # 字典裡面就沒有hobby的屬性了

運行結果:

C:\Users\3-2\PycharmProjects\untitled\venv\Scripts\python.exe C:/Users/3-2/PycharmProjects/untitled/面向對象進階.py
egon
男 男
{'sex': '', 'age': 30, 'name': 'egon'}

Process finished with exit code 0

 

這種用中括弧就可以直接調的方式,比如用字典,列表的實現過程,就是內部存在了item系列這個機制的原因
這種用中括弧就可以直接調的方式,比如用字典,列表的實現過程,就是內部存在了item系列這個機制的原因
object原生支持__delattr__所以才可以直接del f.hobby而不報錯,但是del f['hobby']得通過自己實現,
所以當類方法裡面沒有__delitem__的時候就會報錯

 

二、__new__
__init__:初始化方法
__new__:構造方法,創建一個對象。self就是__new__構造出來的,即__new__方法是self產生的機制
平時是不需要用到執行__new__方法的,以下例子只是簡單說明它是怎麼用的:
class A:
    def __init__(self):
        self.x = 1
        print('in init function')
    def __new__(cls, *args, **kwargs):
        # 傳入一個預設參數cls,執行__new__方法前還沒有self,所以只能傳一個類進來
        print('in new function')
        return object.__new__(A,*args,**kwargs)
        # object.__new__創造了一個新的對象,然後將這個對象傳給self的位置,所以當執行self的時候就可以使用對象了

a = A()  # 實例化,會先執行__new__方法,再執行 __init__方法

運行結果:

in new function
in init function

Process finished with exit code 0
一個典型的設計模式(23種):單例模式
單例模式:一個類始終只有一個實例;當第一次實例化這個類的時候就創建一個實例化的對象;當之後再來實例化的時候,
就會用之前創建的對象

 

# 實現單例模式的例子:

class A: __instance = False # 私有的靜態變數 不希望別人可以使用,必須得經過自己的設置 def __init__(self,name,age): self.name = name self.age = age def __new__(cls, *args, **kwargs): if cls.__instance: # 如果為真就執行下麵代碼,否則執行後面的代碼 # 第二次進來的時候就符合這個了 return cls.__instance # 第二次進來就直接將之前創建的對象返回給self了 cls.__instance = object.__new__(A) # 因為第一次進來的時候就是__instance = False,所以執行這行代碼代碼 # 用object.__new__創建一個類A的新的對象,並且賦值給 cls.__instance return cls.__instance # 這裡就是將新建的對象return回去 egon = A('egg',38) # 真正能實例化對象並且占用記憶體的是object裡面的self,但是這裡使用的對象始終是object.__new__創建的, # 因為自己就有對象了,就不會去使用object裡面的了 # 反正能夠實現單例化的原因是__new__方法的使用 egon.cloth = '小花襖' nezha = A('nazha',25) nezha.shoes = '小白鞋' print(nezha) print(egon) # 執行到這裡根據運行結果顯示記憶體地址是同一個,也就是說第二次實例化的時候是在對第一個實例化後的 # 對象進行操作的,而並沒有再次創建另一個占記憶體的對象,如果第二次實例化傳的參數和原對象一致, # 參數值就會進行覆蓋,如果第二次實例化傳的參數只是原屬性的一部分,則相同的覆蓋,原來的繼續會 # 在表現在現有對象中 print(nezha.name) print(egon.name) # 執行到這裡原來egon的名字已經被nezha覆蓋了 print(nezha.cloth) # 執行到這裡原來egon的cloth會繼續穿在nezha上

運行結果:

<__main__.A object at 0x0000025C0C2293C8>
<__main__.A object at 0x0000025C0C2293C8>
nazha
nazha
小花襖

 

 

三、__hash__

# 在沒有定義__hash__方法的時候,hash都是針對記憶體地址的,而不是針對對象屬性,記憶體地址不一樣,hash的結果也不一樣 

class A:
  def __init__(self,name,sex):
    self.name
= name
a
= A('egn','')
b
= A('egon','nv')
print(hash(a))
print(hash(b))

運行結果:

154259419512
154259419617

 

# 定義了—__hash__方法後,屬性不同hash值也會不同,屬性相同hash值也會相同:
class A:
    def __init__(self,name,sex):
        self.name = name
        self.sex = sex
    def __hash__(self):
        return hash(self.name+self.sex)

a = A('egon','')
b = A('egon','')
c = A('egon','nv')
print(hash(a))
print(hash(b))
print(hash(c))

運行結果:

8385798543724353936
8385798543724353936
-7270162062837990016

 

四、__eq__

 

沒有__eq__方法的時候,兩者比較是比較記憶體地址:
class A:
    def __init__(self,name):
        self.name = name

obj1 = A('egg')
obj2 = A('egg')
print(obj1 == obj2)
# 沒有定義__eq__方法的時候,比較時候預設比較記憶體地址,上面兩個記憶體地址是不一樣的

運行結果:

False

 

# 定義__eq__方法時可以自己設定執行內容,‘==’觸發的_eq_方法
class A:
    def __init__(self,name):
        self.name = name

    def __eq__(self, other):
        if self.name == other.name:
            return True
        else:
            return False

obj1 = A('egg')
obj2 = A('egg')
obj3 = A('EGG')
print(obj1 == obj2)  # 等號觸發的__eq__
print(obj2 == obj3)  # 等號觸發的__eq__

 

運行結果:

True
False

Process finished with exit code 0

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

-Advertisement-
Play Games
更多相關文章
  • Redis支持五種數據類型:string(字元串),hash(哈希),list(列表),set(集合)及zset(sorted set:有序集合)。 String(字元串) string 是 redis 最基本的類型,你可以理解成與 Memcached 一模一樣的類型,一個 key 對應一個 val ...
  • Laravel 中一些常用的小技巧,說不定你就用上了。 1.側欄 網站一般都有側欄,用來顯示分類,標簽,熱門文章,熱門評論啥的,但是這些側欄都是相對獨立的模塊,如果在每一個引入側欄的視圖中都單獨導入與視圖有關的數據的話,未免太冗餘了。所以最佳的做法是:新建一個widgets視圖文件夾,再利用Lara ...
  • django中使用content_type表進行存儲app與model的關係。在permission表中設立了name(許可權的名字,中英文均可)content_type_id(與content_type是外鍵關係,用於綁定model)codename(在寫代碼時使用的許可權名)那麼如何自定義許可權呢? ...
  • [TOC] 項目地址:https://github.com/gatsbyd/melon 介紹 開發服務端程式的一個基本任務是處理併發連接,現在服務端網路編程處理併發連接主要有兩種方式: 1. 當“線程”很廉價時,一臺機器上可以創建遠高於CPU數目的“線程”。這時一個線程只處理一個TCP連接,通常使用 ...
  • 1、概述、 AutoGenerator 是 MyBatis-Plus 的代碼生成器,通過 AutoGenerator 可以快速生成 Entity、Mapper、Mapper XML、Service、Controller 等各個模塊的代碼,極大的提升了開發效率。 2、使用教程 2.1 相關依賴 MyB ...
  • Python2預設解釋器的編碼:ascii; Python3預設解釋器的編碼:UTF-8 ascii碼:只會識別英文字母、數字和標點。8位表示一個英文字元,1個位元組 萬國碼Uicode:目前的所有語言文字,32位,4個位元組 utf-8:壓縮的萬國碼。16位,此編碼中,3位元組表示一個漢字 ...
  • Window 下安裝 Django 如果你還未安裝Python環境需要先下載Python安裝包。 1、Python 下載地址:https://www.python.org/downloads/ 2、Django 下載地址:https://www.djangoproject.com/download/ ...
  • 一.代碼 `點個贊唄` ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...