亚洲综合原千岁中文字幕_国产精品99久久久久久久vr_无码人妻aⅴ一区二区三区浪潮_成人h动漫精品一区二区三

主頁(yè) > 知識(shí)庫(kù) > Python基礎(chǔ)詳解之描述符

Python基礎(chǔ)詳解之描述符

熱門標(biāo)簽:荊州云電銷機(jī)器人供應(yīng)商 外呼不封號(hào)系統(tǒng) 電信營(yíng)業(yè)廳400電話申請(qǐng) 悟空智電銷機(jī)器人6 溫州旅游地圖標(biāo)注 遼寧400電話辦理多少錢 蘇州電銷機(jī)器人十大排行榜 江蘇房產(chǎn)電銷機(jī)器人廠家 幫人做地圖標(biāo)注收費(fèi)算詐騙嗎

一、描述符定義

描述符是一種類,我們把實(shí)現(xiàn)了__get__()、__set__()和__delete__()中的其中任意一種方法的類稱之為描述符。

描述符的作用是用來(lái)代理一個(gè)類的屬性,需要注意的是描述符不能定義在被使用類的構(gòu)造函數(shù)中,只能定義為類的屬性,它只屬于類的,不屬于實(shí)例,我們可以通過(guò)查看實(shí)例和類的字典來(lái)確認(rèn)這一點(diǎn)。

描述符是實(shí)現(xiàn)大部分Python類特性中最底層的數(shù)據(jù)結(jié)構(gòu)的實(shí)現(xiàn)手段,我們常使用的@classmethod、@staticmethd、@property、甚至是__slots__等屬性都是通過(guò)描述符來(lái)實(shí)現(xiàn)的。它是很多高級(jí)庫(kù)和框架的重要工具之一,是使用到裝飾器或者元類的大型框架中的一個(gè)非常重要組件。

如下示例一個(gè)描述符及引用描述符類的代碼:

class Descriptors:
 
    def __init__(self, key, value_type):
        self.key = key
        self.value_type = value_type
 
    def __get__(self, instance, owner):
        print("===> 執(zhí)行Descriptors的 get")
        return instance.__dict__[self.key]
 
    def __set__(self, instance, value):
        print("===> 執(zhí)行Descriptors的 set")
        if not isinstance(value, self.value_type):
            raise TypeError("參數(shù)%s必須為%s" % (self.key, self.value_type))
        instance.__dict__[self.key] = value
 
    def __delete__(self, instance):
        print("===>  執(zhí)行Descriptors的delete")
        instance.__dict__.pop(self.key)
 
 
class Person:
    name = Descriptors("name", str)
    age = Descriptors("age", int)
 
    def __init__(self, name, age):
        self.name = name
        self.age = age
 
 
person = Person("xu", 15)
print(person.__dict__)
person.name
person.name = "xu-1"
print(person.__dict__)
# 結(jié)果:
#     ===> 執(zhí)行Descriptors的 set
#     ===> 執(zhí)行Descriptors的 set
#     {'name': 'xu', 'age': 15}
#     ===> 執(zhí)行Descriptors的 get
#     ===> 執(zhí)行Descriptors的 set
#     {'name': 'xu-1', 'age': 15}

其中,Descriptors類就是一個(gè)描述符,Person是使用描述符的類。

類的__dict__屬性是類的一個(gè)內(nèi)置屬性,類的靜態(tài)函數(shù)、類函數(shù)、普通函數(shù)、全局變量以及一些內(nèi)置的屬性都是放在類__dict__里。

在輸出描述符的變量時(shí),會(huì)調(diào)用描述符中的__get__方法,在設(shè)置描述符變量時(shí),會(huì)調(diào)用描述符中的__set__方法。

二、描述符的種類和優(yōu)先級(jí)

描述符分為數(shù)據(jù)描述符和非數(shù)據(jù)描述符。

至少實(shí)現(xiàn)了內(nèi)置__set__()和__get__()方法的描述符稱為數(shù)據(jù)描述符;實(shí)現(xiàn)了除__set__()以外的方法的描述符稱為非數(shù)據(jù)描述符。

描述符的優(yōu)先級(jí)的高低順序:類屬性 > 數(shù)據(jù)描述符 > 實(shí)例屬性 > 非數(shù)據(jù)描述符 > 找不到的屬性觸發(fā)__getattr__()。

在上述“描述符定義”章節(jié)的例子中,實(shí)例person的屬性優(yōu)先級(jí)低于數(shù)據(jù)描述符Descriptors,所以在賦值或獲取值過(guò)程中,均調(diào)用了描述符的方法。

class Descriptors:
    def __get__(self, instance, owner):
        print("===> 執(zhí)行 Descriptors get")
 
    def __set__(self, instance, value):
        print("===> 執(zhí)行 Descriptors set")
 
    def __delete__(self, instance):
        print("===> 執(zhí)行 Descriptors delete")
 
 
class University:
    name = Descriptors()
 
    def __init__(self, name):
        self.name = name

類屬性 > 數(shù)據(jù)描述符

# 類屬性 > 數(shù)據(jù)描述符
# 在調(diào)用類屬性時(shí),原來(lái)字典中的數(shù)據(jù)描述法被覆蓋為 XX-XX
print(University.__dict__)  # {..., 'name': __main__.Descriptors object at 0x7ff8c0eda278>,}
 
University.name = "XX-XX"
print(University.__dict__)  # {..., 'name': 'XX-XX',}

數(shù)據(jù)描述符 > 實(shí)例屬性

# 數(shù)據(jù)描述符 > 實(shí)例屬性
# 調(diào)用時(shí)會(huì)現(xiàn)在實(shí)例里面找,找不到name屬性,到類里面找,在類的字典里面找到 'name': __main__.Descriptors object at 0x7fce16180a58>
# 初始化時(shí)調(diào)用 Descriptors 的 __set__; un.name 調(diào)用  __get__
print(University.__dict__)
un = University("xx-xx")
un.name
# 結(jié)果:
#     {..., 'name': __main__.Descriptors object at 0x7ff8c0eda278>,}
#     ===> 執(zhí)行 Descriptors set
#     ===> 執(zhí)行 Descriptors get

下面是測(cè)試 實(shí)例屬性、 非數(shù)據(jù)描述符、__getattr__ 使用代碼

class Descriptors:
    def __get__(self, instance, owner):
        print("===>2 執(zhí)行 Descriptors get")
 
 
class University:
    name = Descriptors()
 
    def __init__(self, name):
        self.name = name
 
    def __getattr__(self, item):
        print("---> 查找 item = {}".format(item))

實(shí)例屬性 > 非數(shù)據(jù)描述符

# 實(shí)例屬性 > 非數(shù)據(jù)描述符
# 在創(chuàng)建實(shí)例的時(shí)候,會(huì)在屬性字典中添加 name,后面調(diào)用 un2.name 訪問(wèn),都是訪問(wèn)實(shí)例字典中的 name
un2 = University("xu2")
print(un2.name)  # xu    并沒(méi)有調(diào)用到  Descriptors 的 __get__
print(un2.__dict__)  # {'name': 'xu2'}
un2.name = "xu-2"
print(un2.__dict__)  # {'name': 'xu-2'}

非數(shù)據(jù)描述符 > 找不到的屬性觸發(fā)__getattr__

# 非數(shù)據(jù)描述符 > 找不到的屬性觸發(fā)__getattr__
# 找不到 name1 使用 __getattr__
un3 = University("xu3")
print(un3.name1)  # ---> 查找 item = name1

三、描述符的應(yīng)用

使用描述符檢驗(yàn)數(shù)據(jù)類型

class Typed:
    def __init__(self, key, type):
        self.key = key
        self.type = type
 
    def __get__(self, instance, owner):
        print("---> get 方法")
        # print("instance = {}, owner = {}".format(instance, owner))
        return instance.__dict__[self.key]
 
    def __set__(self, instance, value):
        print("---> set 方法")
        # print("instance = {}, value = {}".format(instance, value))
        if not isinstance(value, self.type):
            # print("設(shè)置name的值不是字符串: type = {}".format(type(value)))
            # return
            raise TypeError("設(shè)置{}的值不是{},當(dāng)前傳入數(shù)據(jù)的類型是{}".format(self.key, self.type, type(value)))
        instance.__dict__[self.key] = value
 
    def __delete__(self, instance):
        print("---> delete 方法")
        # print("instance = {}".format(instance))
        instance.__dict__.pop(self.key)
 
 
class Person:
    name = Typed("name", str)
    age = Typed("age", int)
 
    def __init__(self, name, age, salary):
        self.name = name
        self.age = age
        self.salary = salary
 
 
p1 = Person("xu", 99, 100.0)
print(p1.__dict__)
p1.name = "XU1"
print(p1.__dict__)
del p1.name
print(p1.__dict__)
# 結(jié)果:
#     ---> set 方法
#     ---> set 方法
#     {'name': 'xu', 'age': 99, 'salary': 100.0}
#     ---> set 方法
#     {'name': 'XU1', 'age': 99, 'salary': 100.0}
#     ---> delete 方法
#     {'age': 99, 'salary': 100.0}

四、描述符 + 類裝飾器  (給 Person類添加類屬性)

類裝飾器,道理和函數(shù)裝飾器一樣

def Typed(**kwargs):
    def deco(obj):
        for k, v in kwargs.items():
            setattr(obj, k, v)
        return obj
    return deco
 
 
@Typed(x=1, y=2)  # 1、Typed(x=1, y=2) ==> deco   2、@deco ==> Foo = deco(Foo)
class Foo:
    pass
 
 
# 通過(guò)類裝飾器給類添加屬性
print(Foo.__dict__)  # {......, 'x': 1, 'y': 2}
print(Foo.x)

使用描述符和類裝飾器給 Person類添加類屬性

"""
描述符 + 類裝飾器
"""
class Typed:
    def __init__(self, key, type):
        self.key = key
        self.type = type
 
    def __get__(self, instance, owner):
        print("---> get 方法")
        # print("instance = {}, owner = {}".format(instance, owner))
        return instance.__dict__[self.key]
 
    def __set__(self, instance, value):
        print("---> set 方法")
        # print("instance = {}, value = {}".format(instance, value))
        if not isinstance(value, self.type):
            # print("設(shè)置name的值不是字符串: type = {}".format(type(value)))
            # return
            raise TypeError("設(shè)置{}的值不是{},當(dāng)前傳入數(shù)據(jù)的類型是{}".format(self.key, self.type, type(value)))
        instance.__dict__[self.key] = value
 
    def __delete__(self, instance):
        print("---> delete 方法")
        # print("instance = {}".format(instance))
        instance.__dict__.pop(self.key)
 
 
def deco(**kwargs):
    def wrapper(obj):
        for k, v in kwargs.items():
            setattr(obj, k, Typed(k, v))
        return obj
    return wrapper
 
 
@deco(name=str, age=int)
class Person:
    # name = Typed("name", str)
    # age = Typed("age", int)
 
    def __init__(self, name, age, salary):
        self.name = name
        self.age = age
        self.salary = salary
 
 
p1 = Person("xu", 99, 100.0)
print(Person.__dict__)
print(p1.__dict__)
p1.name = "XU1"
print(p1.__dict__)
del p1.name
print(p1.__dict__)
# 結(jié)果:
#     ---> set 方法
#     ---> set 方法
#     {..., 'name': __main__.Typed object at 0x7f3d79729dd8>, 'age': __main__.Typed object at 0x7f3d79729e48>}
#     {'name': 'xu', 'age': 99, 'salary': 100.0}
#     ---> set 方法
#     {'name': 'XU1', 'age': 99, 'salary': 100.0}
#     ---> delete 方法
#     {'age': 99, 'salary': 100.0}

五、利用描述符自定義 @property

class Lazyproperty:
    def __init__(self, func):
        self.func = func
 
    def __get__(self, instance, owner):
        print("===> Lazypropertt.__get__ 參數(shù): instance = {}, owner = {}".format(instance, owner))
        if instance is None:
            return self
        res = self.func(instance)
        setattr(instance, self.func.__name__, res)
        return self.func(instance)
 
    # def __set__(self, instance, value):
    #     pass
 
 
class Room:
 
    def __init__(self, name, width, height):
        self.name = name
        self.width = width
        self.height = height
 
    # @property  # area=property(area)
    @Lazyproperty  # # area=Lazyproperty(area)
    def area(self):
        return self.width * self.height
 
#  @property 測(cè)試代碼
# r = Room("房間", 2, 3)
# print(Room.__dict__)  # {..., 'area': property object at 0x7f8b42de5ea8>,}
# print(r.area)  # 6
 
# r2 = Room("房間2", 2, 3)
# print(r2.__dict__)  # {'name': '房間2', 'width': 2, 'height': 3}
# print(r2.area)
 
# print(Room.area)  # __main__.Lazyproperty object at 0x7faabd589a58>
 
r3 = Room("房間3", 2, 3)
print(r3.area)
print(r3.area)
# 結(jié)果(只計(jì)算一次):
#     ===> Lazypropertt.__get__ 參數(shù): instance = __main__.Room object at 0x7fd98e3757b8>, owner = class '__main__.Room'>
#     6
#     6

六、property 補(bǔ)充

class Foo:
 
    @property
    def A(self):
        print("===> get A")
 
    @A.setter
    def A(self, val):
        print("===> set A, val = {}".format(val))
 
    @A.deleter
    def A(self):
        print("===> del A")
 
 
f = Foo()
f.A         # ===> get A
f.A = "a"   # ===> set A, val = a
del f.A     # ===> del A
 
 
 
class Foo:
 
    def get_A(self):
        print("===> get_A")
 
    def set_A(self, val):
        print("===> set_A, val = {}".format(val))
 
    def del_A(self):
        print("===> del_A")
 
    A = property(get_A, set_A, del_A)
 
 
f = Foo()
f.A         # ===> get_A
f.A = "a"   # ===> set_A, val = a
del f.A     # ===> del_A

到此這篇關(guān)于Python基礎(chǔ)詳解之描述符的文章就介紹到這了,更多相關(guān)Python描述符內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • Python描述符descriptor使用原理解析
  • python實(shí)現(xiàn)裝飾器、描述符
  • 詳解Python中的Descriptor描述符類
  • Python黑魔法Descriptor描述符的實(shí)例解析
  • Python 的描述符 descriptor詳解
  • 解密Python中的描述符(descriptor)
  • 詳解Python描述符的工作原理

標(biāo)簽:喀什 欽州 三沙 宿遷 黃山 濟(jì)南 臺(tái)灣 景德鎮(zhèn)

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Python基礎(chǔ)詳解之描述符》,本文關(guān)鍵詞  Python,基礎(chǔ),詳解,之,描述,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《Python基礎(chǔ)詳解之描述符》相關(guān)的同類信息!
  • 本頁(yè)收集關(guān)于Python基礎(chǔ)詳解之描述符的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    久久久成人影院| 国产成人精品影视| 成人高清护士在线播放| 成人高清免费| 日本特黄特色aa大片免费| 韩国三级视频在线观看| 国产成人精品综合在线| 欧美另类videosbestsex高清| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 免费的黄色小视频| 欧美激情中文字幕一区二区| 国产精品自拍在线| 久草免费在线色站| a级精品九九九大片免费看| 韩国三级视频网站| 久草免费在线观看| 麻豆网站在线免费观看| 国产成人精品综合久久久| 日韩中文字幕在线播放| 国产一区国产二区国产三区| 国产91精品一区| 九九免费精品视频| 欧美国产日韩久久久| 成人在免费观看视频国产| 成人a大片在线观看| 欧美α片无限看在线观看免费| 精品久久久久久影院免费| 免费一级生活片| 韩国妈妈的朋友在线播放| 亚欧成人乱码一区二区| 九九九在线视频| 可以免费看污视频的网站| 国产一区二区精品尤物| 午夜精品国产自在现线拍| 黄视频网站在线免费观看| 精品久久久久久中文字幕2017| 夜夜操天天爽| 97视频免费在线| 韩国三级视频在线观看| 一本伊大人香蕉高清在线观看| 天天色色色| 韩国毛片基地| 日韩男人天堂| 免费的黄视频| 天天色色色| 一级女性全黄生活片免费| 美女免费毛片| 久久久久久久免费视频| 你懂的在线观看视频| 国产高清视频免费观看| 一级女性全黄生活片免费| 国产成人精品在线| 精品视频一区二区三区免费| 黄视频网站在线免费观看| 精品视频在线观看免费| 精品国产三级a| 日韩男人天堂| 欧美a免费| 国产不卡在线观看视频| 久久国产精品只做精品| 成人av在线播放| 可以免费在线看黄的网站| 欧美大片aaaa一级毛片| 国产91视频网| 91麻豆精品国产自产在线观看一区| 青草国产在线| 午夜精品国产自在现线拍| 国产不卡福利| 韩国毛片免费大片| 你懂的福利视频| 尤物视频网站在线观看| 91麻豆精品国产综合久久久| 精品视频在线观看免费| 91麻豆精品国产自产在线| 免费一级片网站| 韩国毛片 免费| 国产91精品系列在线观看| 日韩一级精品视频在线观看| 国产欧美精品| 精品国产一区二区三区免费| 日韩欧美一二三区| 亚洲 欧美 91| 国产精品免费精品自在线观看| 国产伦久视频免费观看视频| 亚欧乱色一区二区三区| 精品国产香蕉伊思人在线又爽又黄| 欧美激情一区二区三区视频高清| 超级乱淫伦动漫| 国产一区二区精品久久91| 日本特黄特色aa大片免费| 国产91精品系列在线观看| 国产不卡在线看| 黄视频网站在线免费观看| 美国一区二区三区| 日韩免费片| 深夜做爰性大片中文| 黄视频网站免费观看| a级黄色毛片免费播放视频| 91麻豆高清国产在线播放| 国产不卡精品一区二区三区| 91麻豆精品国产高清在线| 一级片片| 亚洲 欧美 91| 久久福利影视| 日韩免费在线视频| 成人免费高清视频| 国产韩国精品一区二区三区| 精品国产一区二区三区国产馆| 可以免费看污视频的网站| 美女免费精品高清毛片在线视| 国产一区精品| 麻豆网站在线看| 国产成人精品影视| 香蕉视频三级| 美女免费毛片| 九九免费精品视频| 成人高清视频在线观看| 九九免费精品视频| 日韩av片免费播放| a级毛片免费观看网站| 国产一区二区精品| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 亚洲第一色在线| 国产91精品系列在线观看| 欧美大片aaaa一级毛片| 午夜家庭影院| 亚洲天堂在线播放| 国产麻豆精品高清在线播放| 麻豆系列 在线视频| 久久99中文字幕| 久久久成人网| 国产一区免费观看| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 九九久久国产精品| 亚洲女人国产香蕉久久精品| 免费毛片基地| 国产一区二区高清视频| 日韩av东京社区男人的天堂| 欧美a级成人淫片免费看| 日本乱中文字幕系列| 国产一区免费观看| 青青久久网| 欧美a级成人淫片免费看| 免费毛片播放| 国产成人精品影视| 久久精品人人做人人爽97| 可以免费在线看黄的网站| 国产一级强片在线观看| 久久国产精品只做精品| 韩国三级视频网站| 国产91视频网| 日本伦理片网站| 国产高清视频免费| 久久久成人网| 精品国产香蕉伊思人在线又爽又黄| 国产不卡在线播放| 四虎影视库国产精品一区| 九九久久国产精品| 一本高清在线| 国产精品自拍亚洲| 九九精品在线| 午夜在线观看视频免费 成人| 色综合久久天天综合绕观看| 日韩专区亚洲综合久久| 精品久久久久久中文字幕一区 | 欧美激情一区二区三区视频高清| 毛片成人永久免费视频| 亚欧成人毛片一区二区三区四区| 色综合久久天天综合观看| 午夜欧美成人久久久久久| 国产精品免费久久| 免费毛片播放| 久久国产一久久高清| 久久成人亚洲| 日本免费看视频| 黄色免费网站在线| 日韩专区在线播放| 国产成人啪精品| 日韩一级黄色片| 99热热久久| 九九热国产视频| 国产91精品系列在线观看| 国产a网| 精品久久久久久中文字幕一区 | 国产91素人搭讪系列天堂| 日韩免费在线视频| 精品视频在线观看免费| 日韩男人天堂| 欧美另类videosbestsex视频| 欧美电影免费看大全| 天天做日日爱| 九九久久国产精品大片| 国产一级生活片| 国产精品自拍亚洲| 久草免费资源| 国产一区国产二区国产三区| 日本伦理片网站| 成人免费一级纶理片| 人人干人人草| 日本特黄一级|