关于隐藏属性
引子:
当类的属性或者类实例对象的属性隐藏的时候必须通过存取器方法来获取和设置这些隐藏的属性。
例如:
def get_name(self,name): #存取器方法
self._name=name
通过外部调用实例函数传参达到获取和设置隐藏属性的目的。这样的函数叫做存取器
缺点:是必须通过调用存取器方法来实现这些功能
为此python提供了更强大的特性函数,即property函数
(1)property()函数
例如:
name=property(get_name,set_name)
property函数封装了get_name,set_name存取器方法,在类的外部可以像其他没隐藏的属性一样操作属性
实际上,property是一个类,它的实例包含了一些魔法方法,上述例子name=property()就是它的一个实例对象,这个实例对象实现了隐藏属性的反隐藏
property还可以通过修饰器的形式实现这个功能
(2)@property装饰器
实现其他语言所拥有的getter,setter和deleter的功能(例如IOS苹果开发,实现获取,设置,删除隐藏的属性)
作用:
用来模拟一个属性
(其实‘模拟’这个说法不是很准确,官方这么说就这么用了,应该说重写更合适,因为模拟前后的属性是同一个属性,用id()函数获取的地址值是相同的)
通过@property装饰器可以对模拟属性的取值和赋值加以控制
例如:
__双下划线隐藏的属性可以被控制在外部修改
@property (实现getter取值)
def score(self):
return self.__score 返回隐藏的属性
(score看似是一个实例函数,其实是一个属性,因为它已经被property装饰器装饰了)
@score.setter (实现设置值,必须写成setter,因为python就是这么规定的)
def score(self,v):
assert 0<=v<=100,‘成绩不合法’
self.__score=v
@score.deleter
def.score(self): (实现删除值)
del self.__socre
注意:
属性一旦删除,就无法设置和获取
为什么用properyt函数?
property通常用在类或者子类中对隐藏属性的改写,类的代码可能有几千行,隐藏属性在内部调用的次数可能就有几百次,如果一个个修改很麻烦,用存取器方法和property函数可以很方便的实现隐藏属性的反隐藏
另外python还有一种处理隐藏属性的方法,不过这个知识点比较晦涩难懂
(3)python的类内部有__getattr__,__setattr__,__delattr__魔法方法,能拦截所有的类属性
例如:
a.name
当在类的外部访问不到隐藏的属性时候,__getattr__会被自动调用,它搜索所的类属性,直到找到隐藏的属性
在类内__getattr__这些魔法方法是可以重写的(重写还是比较复杂的)
class A;
def __getattr__(self,name): #这里的形参name是属性名的意思,不是类属性self.name的那个name
if name=‘name’:
return self.__name
else:
raise AttributeError()
这里要写一个判断,目的是为了当找不到属性时候(类里面根本就没有这个属性,无论是否是隐藏的属性),就直接报错,避免__getattr__一直不停地搜寻下去,形成死循环
在交互模式下用help(list.__getattribute__)查看__getattr__或者__getattribute__的工作原理如下:
__getattribute__(self, name, /)
Return getattr(self, name)
它是返回python的内建函数getattr(),即不断地查询类内的所有属性
class A;
def __setattr__(self,name,value):
if name=‘name’:
self.__name=name
else:
self.__dict__[name]=value
这里设置判断,如果没有这个属性,就直接用__dict__[name]=name来给类实例增加一个属性,避免死循环
所以总结一下,用魔法方法__getattr__这类方法比用property属性方法相比更复杂点,建议处理同类问题的时候用property属性方法更好