# coding=utf-8
#用描述符对属性进行访问控制
class TypedProperty(object):
def __init__(self,name,type_,default=None): #描述符实例属性中保存“_特性变量名(name/num...)”、类型、默认值
self.name="_"+name
self.type=type_
self.default=type_() if default is None else default
def __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...)=value
def __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.name
f.num=5
print(f.num)
'''
nobody
0
Guido
5
'''
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__(...)
花了几天时间,初步理解了。