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

分类: 计算机技术
推荐阅读:
Python使用莱布尼茨公式计算圆周率 可以通过莱布尼茨级数的求和来逼近圆周率;公式为:π/4 = 1 - 1/3 + 1/5 - 1/7 + 1/9 - ......(无限项)
Python中__getitem__()方法和索引器[]的详细用法 在本文中将详细描述如何使用Python为自定义对象使用索引器和类的__getitem__()方法;
Rust实现冒泡排序算法(Bubble Sort) 本文将使用Rust语言实现冒泡排序算法;
System has not been booted with systemd as init system (PID 1). Can't operate.解决方法 在WSL(Windows Subsystem for Linux,适用于Linux的Windows子系统)下通过systemctl命令启动某些服务将造成System has not been booted with systemd as init system (PID 1). Can't operate.这样的错误;
Requests详细教程 Requests 是一个功能强大、优雅而简单的 Python HTTP库;使用Requests发送网络请求整个过程将变得非常简单。
Python使用集合实现内容去重 有这么一个场景,某个文件用于记录人名(也可以是URL或者其他的),编辑发现,出现了重复的内容。那么我们如何使用Python来进行去重呢?