初学python描述符

来源:互联网 发布:四维排课软件免费下载 编辑:程序博客网 时间:2024/05/17 03:14
描述符:当某属性被访问时,你也许想创建一个日志记录,而最好的方法编写一个执行符合需要的动作函数,然后指定它在属性被访问时运行。


property---把函数的调用伪装成对属性的访问
例子:
class User(object):def __init__(self,name,age):self.name=nameself.age=agedef change_age(self,age):if age < 0:raise ValueError("Age should larger than 0") self.age=age以上代码中,运行a=User("Tom",12)a.age //12a.change_age(-20) //"Age should larger than 0"a.age=13a.age  //13


利用property之后可以改成:
class User(object):def __init__(self,name,age):self.name=nameself.age=age@propertydef age(self):#do something...return self._age@age.setterdef age(self,value):if value < 0:raise ValueError("Age should larger than 0")self._age=value以上代码中,运行a=User("Tom",12)a.age //12a.age=-20 //"Age should larger than 0"   (把函数的调用伪装成对属性的访问)a.age=13a.age  //13


对property来说,最大的缺点就是它们不能重复使用。就是说如果你多几个类似age的属性时,你需要每个都设置property和相应的setter,这样就非常繁琐。虽然property可以让类从外部看起来接口整洁漂亮,但是却做不到内部同样整洁漂亮。描述符就是解决此类问题的关键技术。当某属性被访问时,你也许想创建一个日志记录,而最好的方法编写一个执行符合需要的动作函数,然后指定它在属性被访问时运行。

利用描述符的情况:
#coding=utf-8from weakref import WeakKeyDictionaryclass NonNegative(object):def __init__(self,default):self.default = default   #默认值self.data = WeakKeyDictionary() #为了防止内存泄漏,不想仅仅因为它在描述符的字典中就让一个无用的实例一直存活着def __get__(self,instance,owner):return self.data.get(instance,self.default) #给出字典中instance的值,如果instance为空则返回默认值def __set__(self,instance,value):if value < 0:raise ValueError("Negative value not allowed")self.data[instance] = value  #设置instance键值class User(object):age = NonNegative(0)  #创建NonNegative的一个实例名为age,作为一个NonNegative对象,这里的描述符一定要放在类的层级上。score = NonNegative(0)def __init__(self,name,age,score):self.name=nameself.age=ageself.score = scoreu = User("Tom",12,34)print u.age   # calls User.age.__get__(u,User)u.score = 45  # calss User.score.__set__(u,45)print u.score   # calls User.score.__get__(u,User)


这与我们平时的User的各个实例的维护方式不太相同。在这里,我们的age作为一个实例,其中的data字典中用来存储不同User实例的age的值。
描述符可以用来做什么——它们提供了一种方法将property的逻辑隔离到单独的类中来处理。
0 0