实例的属性有些是需要进行约束的,比如分数、年龄都需要是整形,而不能是浮点数类型,因此为了约束这些属性的取值,可以在类的定义中定义赋值和取值函数,在赋值函数中对变量的取值进行约束,如下所示
class Student(object):
def get_score(self):
return self._score
def set_score(self, value):
if not isinstance(value, int):
raise ValueError('score must be an integer!')
if value < 0 or value > 100:
raise ValueError('score must between 0 ~ 100!')
self._score = value
但是这么做每次对变量赋值都需要调用函数,能否不调用函数,而直接像使用属性一样取值赋值呢,答案是可以的,就需要用到@property
class Junior(object):
def __init__(self, name, score, sex, height, weight):
self.name = name
self.score = score
self.sex = sex
self.height = height
self.weight = weight
@property
def birth(self):
return self._birth
@birth.setter
def birth(self,year):
if not isinstance(year,int):
raise valueerror('not integer')
self._birth=year
@property
def age(self):
return 2017-self._birth
s=Junior('kimi',96,'M',180,65)
s.birth=1989
print(s.birth)
print(s.age)
上例中就可以将birth当做属性来调用赋值,又可以对它进行约束,这个例子中birth属于可读可写,而age只有@property,没有setter,因此只是可读,不能赋值。
作业:
class Screen(object):
@property
def width(self):
return self._width
@property
def height(self):
return self._height
@width.setter
def width(self,value):
if not isinstance(value,int):
raise error('not an integer')
self._width=value
@height.setter
def height(self, value):
if not isinstance(value, int):
raise error('not an integer')
self._height = value
@property
def resolution(self):
self._resolution=self.width*self.height
return self._resolution
# test:
s = Screen()
s.width = 1024
s.height = 768
print(s.resolution)
assert s.resolution == 786432, '1024 * 768 = %d ?' % s.resolution
取值和赋值函数内部的变量不能与函数名相同,一定要有所区分,不然会被认为是函数名,导致出问题,因此函数名为width,函数内部的变量名为self._width
以下做了一个实验:
class Dog(object):
__slots__=('_name','_spec','_birth')
def __init__(self,name,spec='Unknown'):
self._name=name
self._spec=spec
@property
def _age(self):
return 2017-self._birth
@property
def birth(self):
return self._birth
@birth.setter
def birth(self,year):
if not isinstance(year,int):
raise Error('Wrong')
else:
self._birth=year
dd=Dog('didi','wolf')
dd._birth=2000
print(dd.birth)
dd.birth=2000
print(dd.birth)
print(dd._age)
__slot__针对的是self的属性,不包括方法名,但是方法类的属性名也需要包括进去,比如birth不属于,而_birth就属于约束范围,属性方法birth在函数外调用时用birth名,但是在函数内部属性需要另取他名,否则会和函数名相同,导致反复调用,程序崩溃,上例中可以使用birth和_birth赋值,但是两者走的路径不同,birth是属性方法赋值,而_birth是直接给属性赋值