面向對象的進階(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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...