日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程语言 > python >内容正文

python

python属性和方法的区别_Python中几种属性访问的区别

發(fā)布時(shí)間:2023/12/2 python 40 豆豆
生活随笔 收集整理的這篇文章主要介紹了 python属性和方法的区别_Python中几种属性访问的区别 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

起步

python的提供一系列和屬性訪問有關(guān)的特殊方法:__get__, __getattr__, __getattribute__, __getitem__ 。本文闡述它們的區(qū)別和用法。

屬性的訪問機(jī)制

一般情況下,屬性訪問的默認(rèn)行為是從對象的字典中獲取,并當(dāng)獲取不到時(shí)會沿著一定的查找鏈進(jìn)行查找。例如 a.x 的查找鏈就是,從 a.__dict__['x'] ,然后是 type(a).__dict__['x'] ,再通過 type(a) 的基類開始查找。

若查找鏈都獲取不到屬性,則拋出 AttributeError 異常。

__getattr__ 方法

這個(gè)方法是當(dāng)對象的屬性不存在是調(diào)用。如果通過正常的機(jī)制能找到對象屬性的話,不會調(diào)用 __getattr__ 方法。

class A:

a = 1

def __getattr__(self, item):

print('__getattr__ call')

return item

t = A()

print(t.a)

print(t.b)

# output

1

__getattr__ call

b

__getattribute__ 方法

這個(gè)方法會被無條件調(diào)用。不管屬性存不存在。如果類中還定義了 __getattr__ ,則不會調(diào)用 __getattr__() 方法,除非在 __getattribute__ 方法中顯示調(diào)用__getattr__() 或者拋出了 AttributeError 。

class A:

a = 1

def __getattribute__(self, item):

print('__getattribute__ call')

raise AttributeError

def __getattr__(self, item):

print('__getattr__ call')

return item

t = A()

print(t.a)

print(t.b)

所以一般情況下,為了保留 __getattr__ 的作用,__getattribute__() 方法中一般返回父類的同名方法:

def __getattribute__(self, item):

return object.__getattribute__(self, item)

使用基類的方法來獲取屬性能避免在方法中出現(xiàn)無限遞歸的情況。

__get__ 方法

這個(gè)方法比較簡單說明,它與前面的關(guān)系不大。

如果一個(gè)類中定義了 __get__(), __set__() 或 __delete__() 中的任何方法。則這個(gè)類的對象稱為描述符。

class Descri(object):

def __get__(self, obj, type=None):

print("call get")

def __set__(self, obj, value):

print("call set")

class A(object):

x = Descri()

a = A()

a.__dict__['x'] = 1 # 不會調(diào)用 __get__

a.x # 調(diào)用 __get__

如果查找的屬性是在描述符對象中,則這個(gè)描述符會覆蓋上文說的屬性訪問機(jī)制,體現(xiàn)在查找鏈的不同,而這個(gè)行文也會因?yàn)檎{(diào)用的不同而稍有不一樣:

如果調(diào)用是對象實(shí)例(題目中的調(diào)用方式),a.x 則轉(zhuǎn)換為調(diào)用: 。type(a).__dict__['x'].__get__(a, type(a))

如果調(diào)用的是類屬性, A.x 則轉(zhuǎn)換為:A.__dict__['x'].__get__(None, A)

其他情況見文末參考資料的文檔

__getitem__ 方法

這個(gè)調(diào)用也屬于無條件調(diào)用,這點(diǎn)與 __getattribute__ 一致。區(qū)別在于 __getitem__ 讓類實(shí)例允許 [] 運(yùn)算,可以這樣理解:

__getattribute__ 適用于所有 . 運(yùn)算符;

__getitem__ 適用于所有 [] 運(yùn)算符。

class A(object):

a = 1

def __getitem__(self, item):

print('__getitem__ call')

return item

t = A()

print(t['a'])

print(t['b'])

如果僅僅想要對象能夠通過 [] 獲取對象屬性可以簡單的:

def __getitem(self, item):

return object.__getattribute__(self, item)

總結(jié)

當(dāng)這幾個(gè)方法同時(shí)出現(xiàn)可能就會擾亂你了。我在網(wǎng)上看到一份示例還不錯(cuò),稍微改了下:

class C(object):

a = 'abc'

def __getattribute__(self, *args, **kwargs):

print("__getattribute__() is called")

return object.__getattribute__(self, *args, **kwargs)

# return "haha"

def __getattr__(self, name):

print("__getattr__() is called ")

return name + " from getattr"

def __get__(self, instance, owner):

print("__get__() is called", instance, owner)

return self

def __getitem__(self, item):

print('__getitem__ call')

return object.__getattribute__(self, item)

def foo(self, x):

print(x)

class C2(object):

d = C()

if __name__ == '__main__':

c = C()

c2 = C2()

print(c.a)

print(c.zzzzzzzz)

c2.d

print(c2.d.a)

print(c['a'])

可以結(jié)合輸出慢慢理解,這里還沒涉及繼承關(guān)系呢。總之,每個(gè)以 __ get 為前綴的方法都是獲取對象內(nèi)部數(shù)據(jù)的鉤子,名稱不一樣,用途也存在較大的差異,只有在實(shí)踐中理解它們,才能真正掌握它們的用法。

參考

總結(jié)

以上是生活随笔為你收集整理的python属性和方法的区别_Python中几种属性访问的区别的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。