@property装饰器(学生成绩、个人信息的约束)

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.score = score

当我们想要修改一个 Student 的 scroe 属性时,可以这么写:

s = Student('Bob', 59)
s.score = 60

但是也可以这么写:

s.score = 1000

此时score变成1000,显然,直接给属性赋值无法检查分数的有效性。

如果利用两个方法,即get_score() 和 set_score():

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.__score = score
    def get_score(self):
        return self.__score
    def set_score(self, score):
        if score < 0 or score > 100:
            raise ValueError('invalid score')
        self.__score = score

这样一来,s.set_score(1000) 就会报错。

这种使用 get/set 方法来封装对一个属性的访问在许多面向对象编程的语言中都很常见。

但是写 s.get_score() 和 s.set_score() 没有直接写 s.score 来得直接。

有没有两全其美的方法?----有。

因为Python支持高阶函数,在函数式编程中我们介绍了装饰器函数,可以用装饰器函数(即@property、@score.setter)把 get/set 方法“装饰”成属性调用,实例化的对象使用属性时,不是调用属性,而是用的方法名:

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.__score = score
    @property
    def score(self):
        return self.__score
    @score.setter
    def score(self, score):
        if score < 0 or score > 100:
            raise ValueError('invalid score')
        self.__score = score

@property其实就是实现了getter功能;

@property.setter实现的是setter功能;

还有一个 @property.deleter实现删除功能

现在,就可以像使用属性一样设置score了:

>>> s = Student('Bob', 59)
>>> s.score = 60
>>> print s.score
60
>>> s.score = 1000
Traceback (most recent call last):
  ...
ValueError: invalid score

以上说明对 score 赋值实际调用的是 set方法。

如果只实现了 @property(而没有实现@property.setter),那么该属性为“只读”属性。

定义方法的时候 @property必须在 @property.setter之前,且二者修饰的方法名相同

附加任务:

请给Student类加一个grade属性,根据 score 计算 A(>=80)、B、C(<60)。

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.__score = score
    @property
    def score(self):
        return self.__score
    @score.setter
    def score(self, score):
        if score < 0 or score > 100:
            raise ValueError('invalid score')
        self.__score = score
        
    @property
    def grade(self):
        if self.__score >= 80:
            return 'A'
        elif self.__score < 60:
            return 'C'
        else:
            return 'B'
            
>>> s = Student('Bob', 59)
>>> print(s.score)
59
>>> s.score = 82
>>> print(s.score)
82
>>> print(s.grade)
A
>>> s.score = 1000
Traceback (most recent call last):
  ...
ValueError: invalid score

利用@property对属性赋值进行约束!还可参考下例,个人信息的约束:
https://blog.csdn.net/qq_41359051/article/details/82939655

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值