在类的构造中,如果所要求得的结果是名词,但是结果的求值必须通过类的构造方法才能实现,那么可以通过在方法上添加装饰器把方法伪造成类的属性进行打印,这样就符合了名词的要求。
- 把方法伪造成类的属性: @property 伪造的属性名是构造方法的方法名,属性值是构造方法的返回值,使用@property后调用不需要加括号。
Note 注意:凡是类似于bmi这种需要计算并且名字又是名词属性的,都需要在构造方法上添加@property装饰器
'''
计算bmi值,第一种方式:把bmi的作为构造方法,可以实现基本bmi求值
'''
class Person(object):
def __init__(self, name, age, weight, hight):
self.name = name
self.age = age
self.__weight = weight
self.__hight = hight
def bmi(self):
return '%s的bmi值为%s' % (self.name, self.__weight / self.__hight ** 2)
p1 = Person('李镇', 18, 120, 1.77)
print(p1.bmi())
# 这中方式虽然可以实现求值,语法也没有问题,但是bmi是名词,方法确是动词,不是很合适
'''
第二种方式:在bmi构造方法上添加装饰器,把方法转换成为类的属性值
'''
class Person(object):
def __init__(self, name, age, weight, hight):
self.name = name
self.age = age
self.__weight = weight
self.__hight = hight
@property
def bmi(self):
return '%s的bmi值为%s' % (self.name, self.__weight / self.__hight ** 2)
# 在方法上添加@property可以把方法名转换成类的属性
p1 = Person('李镇', 18, 60, 1.77)
print(p1.bmi)
# 转换后直接用实例对象调用属性,不加括号。方法名为伪装后的属性名,方法的结果是属性的值
# 把一个方法伪装成为一个属性,在代码级别上没有本质的提升,但是可以让代码看起来更加合理。
伪造的属性不能直接修改属性值,如果需要修改必须用到@伪造属性名.setter来设定
- 写构造函数:函数名是伪造成属性前的方法名,参数分别是:self和修改值
在改方法上添加@伪造属性名.setter ```python class Student(object): def init(self, name, age, gender): self.name = name if type(age) is int:
self.__age = age
else:
print("请输入正确的数据类型")
可以在init中判断属性是否符合要求
self.gender = gender
@property def age(self): return self.__age
@age.setter
伪装成属性的方法名.setter
def age(self, a): self.__age = a if type(a) is int else print(“请输入正确的数据类型”)
# 伪装前的方法,修改内容
# self.__age = a
s1 = Student(‘刘洋’, 23, ‘女’) print(s1.age) s1.age = 26 print(s1.age)
修改属性成功
- 删除属性值
- 需要用到@属性名.deleter
- 创建执行删除操作的构造方法, del 属性进行删除
```python
class Student(object):
def __init__(self, name, age, gender):
self.name = name
if type(age) is int:
self.__age = age
else:
print("请输入正确的数据类型")
# 可以在__init__中判断属性是否符合要求
self.gender = gender
@property
def age(self):
return self.__age
@age.setter
# 伪装成属性的方法名.setter
def age(self, a):
self.__age = a if type(a) is int else print("请输入正确的数据类型")
# 伪装前的方法,修改内容
# self.__age = a
@age.deleter
# 删除的装饰器
def age(self):
del self.__age
#这里需要在构造方法内执行del self.__age才能真正执行删除的操作
s1 = Student('刘洋', 23, '女')
print(s1.age)
s1.age = 26
print(s1.age)
# 修改属性成功
del s1.age
# del 对象.属性执行删除的命令
'''
结果是:
23
26
Traceback (most recent call last):
File "/Volumes/workspace/python-study/re_st/属性.py", line 80, in <module>
print(s1.age)
File "/Volumes/workspace/python-study/re_st/属性.py", line 59, in age
return self.__age
AttributeError: 'Student' object has no attribute '_Student__age'
此时已经执行完删除的操作,所以再次执行打印的时候会进行报错
'''
Note
- 需要注意的是:伪造前的方法名、修改属性值的装饰器中.setter前的内容和修改属性值的构造方法名必须保持一致
- 在init和其他的构造方法中可以通过if条件来判断属性值是否符合要求,但是不能用return来返回,可以用print或者赋值操作
- 装饰器@property和@伪造属性名.setter是相对应的,前者用于伪造属性,后者用于修改属性值