一、property函数:
@property是python的一种装饰器,是用来修饰方法的;
作用:
使用@property装饰器来创建只读属性,@property装饰器会将方法转换为相同名称的只读属性,可以与所定义的属性配合使用,这样可以防止属性被修改;
使用场景:
1、修饰方法是方法可以像属性一样访问:
class DataSet(object):@propertydef method_with_property(self): #含有@propertyreturn 15def method_without_property(self): #不含@propertyreturn 15l = DataSet()
# 加了@property后,可以用调用属性的形式来调用方法,后面不需要加()
print(l.method_with_property)
print(l.method_without_property())
注意事项:若是采用l.method_with_property()函数方式调用,则会出现错误:TypeError: 'int' object is not callable,也就是说添加@property 后,这个方法就变成了一个属性,如果后面加入了(),那么就是当作函数来调用,而它却不是callable(可调用)的。
2、与所定义的属性配合使用,这样可以防止属性被修改:
由于python进行属性的定义时,没办法设置私有属性,因此要通过@property的方法来进行设置。这样可以隐藏属性名,让用户进行使用的时候无法随意修改:
例1:
class DataSet(object):def __init__(self):self._images = 1self._labels = 2 #定义属性的名称#方法加入@property后,相当于一个属性,这属性让用户使用,但用户无法随意修改。@propertydef images(self): return self._images @propertydef labels(self):return self._labels
#属性调用时,直接调用images即可,而不用知道属性名_images,因此用户无法更改属性,从而保护类的属性。
l = DataSet()
# 加了@property后,可以用调用属性的形式来调用方法,后面不需要加()
print(l.images)
例2:
class Stranger(object):def __init__(self, gender=None, age=None, job=None):self.gender = genderself._age = age # 这里的成员属性_age需要与成员方法age()区分开self.jobb = job# 读取age@property # 实现一个age相关的getter方法def age(self):return self._age# 设置age@age.setter # 实现一个age相关的setter方法def age(self, value):if isinstance(value, int):self._age = valueelse:raise ValueError("'int' type need")if __name__ == "__main__":# 创建一个“妹子”meizi = Stranger()meizi.age = 18 # 使用时注意是.age,不是._ageprint("年龄:{age}".format(age=meizi.age)) # 年龄:18
注意事项:
1、属性名与方法名一定要区分开,不然会进入死循环(self._age,def age())
2、实例化的对象使用属性时,不是调用属性(meizi._age),而是用的方法名(meizi.age)
3、@property其实实现了getter功能; @xxx.setter实现的是setter功能;@xxx.deleter实现删除功能
4、定义方法的时候 @property必须在 @xxx.setter之前,且二者修饰的方法名相同(age())
5、如果只实现了 @property(而没有实现@xxx.setter),那么该属性为 只读属性
例3:另一种方式实现:
利用函数接口:property(fget=None, fset=None, fdel=None, doc=None)
class Stranger(object):def __init__(self, gender=None, age=None, job=None):self.gender = genderself._age = ageself.jobb = job# 设置_agedef set_age(self, age):if isinstance(age, int):self._age = ageelse:raise ValueError("'int' type need")# 读取_agedef get_age(self):return self._age# 使得实例化对象可以利用.age方式来访问age = property(get_age, set_age)if __name__ == "__main__":# 创建一个“妹子”meizi = Stranger()meizi.age = 18print("年龄:{age}".format(age=meizi.age)) # 输出: 年龄:18