# coding=utf-8#用描述符对属性进行访问控制class TypedProperty(object):def __init__(self,name,type_,default=None): #描述符实例属性中保存“_特性变量名(name/num...)”、类型、默认值self.name="_"+nameself.type=type_self.default=type_() if default is None else defaultdef __get__(self,instance,cls):return getattr(instance,self.name,self.default) if instance else self#f._特性变量名(name/num...)def __set__(self,instance,value):if not isinstance(value,self.type):raise TypeError("Must be a %s",self.type)setattr(instance,self.name,value)#f._特性变量名(name/num...)=valuedef __delete__(self,instance):raise AttributeError("Can't delete attribute")class Foo(object):name=TypedProperty("name",str,"nobody")num=TypedProperty("num",int)f=Foo()print(f.name)print(f.num)f.name="Guido"print(f.name)#del f.namef.num=5print(f.num)'''nobody0Guido5'''
TypedProperty()为描述符类,name/num...为描述符类的实例,也为Foo类的类属性(变量),f为Foo的实例
f._name等为f的属性。
f.name 工作时调用Foo.__getattribute__()先从实例f 中查找name,没找到,再从Foo中找,查找到Foo的类属性name,即Foo.name因其是描述符类TypedProperty的实例,解释器将Foo.name 转化为 Foo.__dict__['name'].__get__(...) ,取得f._name的值(真实的值保存在f._name中)。
当f.name=value时,解释器将其 转化为 Foo.__dict__['name'].__set__(...),设置f._name=value。
当del f.name时,解释器将其 转化为 Foo.__dict__['name'].__delete__(...)
花了几天时间,初步理解了。
Python利用描述符进行属性访问控制 完成属性数据类型强制定义(如C语言) 属性读写及删除操作