Python中@property和@*.setter装饰器的详细用法

有这么个场景:

class result:
    def __init__(self):
        self.data = [11,33,44,55,66]

exam_result = result()
exam_result.data = 1

exam_result.data 可以被随意修改,可我只希望它是一个只读属性,应该怎么做呢?

@property

@property装饰器的作用是,将一个函数装饰成为类的一个属性,当某个属性需要动态的计算时,用@property来装饰函数最好不过了;

了解@函数修饰符的人应该知道property实际上是一个函数。

使用方法:

class result:
    def __init__(self):
        self.__data = [11,33,44,55,66]

    @property
    def data(self):
        return self.__data

这时候再对 exam_result.data进行赋值,就会发生AttributeError: can’t set attribute的错误!

这时候data不仅是一个方法,还是类的一个属性,所以我们可以利用这个特性动态的计算某些结果:

    @property
    def total(self):
        total_ = 0
        for value in self.__data:
            total_ += value
        return total_

为result类添加一个新的值total用于计算总分数。

修改代码,实现一个新的功能,获取分数及格的总数;

        self.__pass_score = 60

添加一个 __pass_score的属性,用于记录及格分数线,暂且初始化为60;

    @property
    def passCount(self):
        count = 0
        for value in self.__data:
            if value >= self.__pass_score:
                count+=1
        return count

添加一个passCount函数,用于获取及格的数量。使用@property修饰。

现在我希望能够设置及格分数线,并要保证其值合法;

@*.setter

@*.setter 允许你对已用@property装饰的函数赋值:

    @passScore.setter
    def passScore(self,value):
        if not 0 < value < 100:
            raise ValueError('必须为1~99的数')
        self.__pass_score=value

需要注意,setter装饰器必须在property的后面,且两个被修饰的属性(函数)名称必须保持一致。

因为属性实际是通过函数执行获得,所以我们可以对值进行判断或实现更为复杂的操作。

完整代码:

class result:
    def __init__(self):
        self.__data = [11,33,44,55,66]
        self.__pass_score = 60
    @property
    def data(self):
        return self.__data
    
    @property
    def total(self):
        total_ = 0
        for value in self.__data:
            total_ += value
        return total_
    @property
    def passCount(self):
        count = 0
        for value in self.__data:
            if value >= self.__pass_score:
                count+=1
        return count
    @property
    def passScore(self):
        return self.__pass_score
    @passScore.setter
    def passScore(self,value):
        if not 0 < value < 100:
            raise ValueError('必须为1~99的数')
        self.__pass_score=value

原创内容,如需转载,请注明出处;

本文地址: https://www.perfcode.com/p/966.html

分类: 计算机技术
推荐阅读:
获取Rust代码文件名、行号及位置信息 Rust提供了一组特殊的宏,可以在代码中获取文件名、行号和列号。这些宏分别是file!、line!和column!。通过在代码中插入这些宏,您可以在编译时获取有关代码位置的关键信息,从而更轻松地追踪问题。
Golang实现base64加密解密 Base64是网络上最常见的用于传输8Bit字节码的编码方式之一,Base64就是一种基于64个可打印字符来表示二进制数据的方法。
C语言实现斐波那契数列 对于斐波那契数列,其下一个数字是前两个数字的和,例如:0、1、1、2、3、5、8、13、21等等;斐波那契数列的前两个数为0和1;使用C语言实现斐波那契数列的方法有两种,递归和无递归;
Golang实现判断文件或文件夹是否存在 golang通过使用 os包中的Stat()函数和IsNotExist()函数即可判断文件或文件夹是否存在。
如何禁用VSCode后台自动更新 VSCode是我比较喜欢的IDE了,但它也会偷偷摸摸的下载更新并自动安装,而且频率非常高;我并不太喜欢软件在自己毫不知情的情况下自动完成更新,所以我决定禁止它的这种行为;
C语言中 i++ 和 ++i 的区别 在C语言中,++ 运算符也叫递增运算符,只需要一个操作数,属于一元运算符;本文将讨论前缀++运算符和后缀++运算符的区别,以及符号优先级的问题;