【上海python培训】Python使用@property装饰类方法

时间:2018-02-05 09:46:38   来源:上海尚学堂   阅读:

Python版本:3.5.2

假如我们有一个Student类,并在其中定义了一个score属性,但是score属性会被显露出去,没办法检查参数,导致成绩可以随意更改:

stu = Student()
stu.score = 9999

这显然是不合逻辑的,为了限制score的范围,可以通过一个set_score()方法来设置成绩,并通过一个get_score()方法来获取成绩,这样的话,就可以实现参数的检查:

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

调用及结果如下:

调用:

stu = Student()
stu.set_score(99)
print(stu.get_score())
stu.set_score(9999)

结果:

99

Traceback (most recent call last):
    stu.set_score(9999)
    raise ValueError('score must between 0 ~ 100!')
ValueError: score must between 0 ~ 100!

但是,上面的调用方法又略显复杂,没有直接用属性那么直接简单。

这个问题是可以得到解决的,装饰器可以给函数动态的加上功能,那么对于类方法,一样可以起作用,可以使用@property装饰器将类方法变成属性来调用:

class Student(object):

    @property
    def score(self):
        return self._score

    @score.setter
    def 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

把一个get方法变成属性,只需要加上@property就可以,此时,@property本身又创建了另一个装饰器@score.setter,负责把set方法也变成属性并可以赋值,之后我们就可以直接对score进行属性的操作并带有参数的检查:

调用:

stu = Student()
stu.score = 99
print(stu.score)
stu.score = 9999

结果:

99

Traceback (most recent call last):
    stu.score = 9999
    raise ValueError('score must between 0 ~ 100!')
ValueError: score must between 0 ~ 100!

有了这个@property,我们就可以无顾虑的操作类的属性,上海python培训。

如果我们我们在类中只定义了get方法(并用@property装饰),而没有定义set方法,那么这个属性就会变成只读属性。

@property广泛应用在类的定义中,可以让调用者写出简短的代码,同时保证对参数进行必要的检查,这样,程序运行时就减少了出错的可能性。


上海python培训感谢原文作者Lance# 的分享,原文地址:https://www.cnblogs.com/GyForever1004/p/8414244.html
分享:0

电话咨询

客服热线服务时间

周一至周五 9:00-21:00

周六至周日 9:00-18:00

咨询电话

021-67690939
15201841284

微信扫一扫