欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Python 如何定义只读属性?【新手必学】

程序员文章站 2023-09-28 23:32:31
前言本文的文字及图片来源于网络,仅供学习、交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理。作者:Daniel2333如果还没学到属性问题,看不懂不怪你,可以先去小编的Python交流.裙 :一久武其而而流一思(数字的谐音)转换下可以找到了,里面有最新Python教程项 ......

 

在java里, 若要为一个类定义只读的属性, 只需要将目标属性用private修饰, 然后只提供getter()而不提供setter(). 但python没有private关键字, 如何定义只读属性呢? 有两种方法, 第一种跟java类似, 通过定义私有属性实现. 第二种是通过__setattr__.

通过私有属性

用私有属性+@property定义只读属性, 需要预先定义好属性名, 然后实现对应的getter方法.,如果对属性还不懂。那建议你先去小编的python学习.裙 :一久武其而而流一思(数字的谐音)转换下可以找到了,里面有最新python教程项目,先系统学习下!

class vector2d(object):
    def __init__(self, x, y):
        self.__x = float(x)
        self.__y = float(y)

    @property
    def x(self):
        return self.__x
    @property
    def y(self):
        return self.__y

if __name__ == "__main__":
    v = vector2d(3, 4)
    print(v.x, v.y)
    v.x = 8 # error will be raised.

输出:

(3.0, 4.0)
traceback (most recent call last):
  file ...., line 16, in <module>
    v.x = 8 # error will be raised.
attributeerror: can't set attribute

可以看出, 属性x是可读但不可写的.

通过__setattr__

当我们调用obj.attr=value时发生了什么?

很简单, 调用了obj__setattr__方法. 可通过以下代码验证:

class mycls():
    def __init__(self):
        pass

    def __setattr__(self, f, v):
        print 'setting %r = %r'%(f, v)
if __name__ == '__main__':
    obj = mycls()
    obj.new_field = 1

输出:

setting 'new_field' = 1
  • 1

所以呢, 只需要在__setattr__ 方法里挡一下, 就可以阻止属性值的设置, 可谓是釜底抽薪. 
代码:

# encoding=utf8
class mycls(object):
    readonly_property = 'readonly_property' 
    def __init__(self):
        pass
    def __setattr__(self, f, v):
        if f == 'readonly_property':
            raise attributeerror('{}.{} is read only'.\
                                 format(type(self).__name__, f))

        else:
            self.__dict__[f] = v

if __name__ == '__main__':
    obj = mycls()

    obj.any_other_property = 'any_other_property'
    print(obj.any_other_property)

    print(obj.readonly_property)
    obj.readonly_property = 1

输出:

any_other_property
readonly_property
traceback (most recent call last):
  file "...", line 21, in <module>
    obj.readonly_property = 1
    ...
  attributeerror: mycls.readonly_property is read onl