python防止随意修改类属性

如果不想允许随意修改一个类的某个属性,常用的方法是使用property装饰器以及在属性前加下划线。

class V:
    def __init__(self, x): 
        self._x = x
    
    @property
    def x(self):
        return self._x 
    

虽然这样是没法直接修改x了,但还是可以通过_x很轻易地修改x

>>>v = V(5)
>>>v.x
5
>>>v._x
5
>>>v.x = 4
Traceback (most recent call last):
  File "", line 1, in 
AttributeError: cant set attribute
>>>v._x = 4
>>>v.x
4

万一哪天手抖了呢...
但如果使用双下划线——

class V:
    def __init__(self,x):
        self.__x = x

    @property
    def x(self):
        return self.__x
>>>v = V(5)
>>>v.__x = 4
>>>v.__x
4
>>>v.x
5

原因在于python的名称改写特性(name mangling)。python会将以双下划线开头的实例属性名前加上一个下划线和类名存入实例的__dict__属性中。

>>>v = V(5)
>>>v.__dict__
{'_V__x': 5}
>>>v.__x = 4
>>>v.__dict__
{'_V__x': 5, '__x': 4}

当然,修改v.x的值还是可以做到的的,只需通过修改v._V__x就可以了,不过误操作的可能性已经降低很多了。

p.s.类也有__dict__属性。

然而名称改写的应用并不只是防止无意访问这么简单。根据Effective Python:

只有一种情况是可以合理使用private属性的,那就是避免子类的属性名与超类相冲突。(详见P79-P80)

你可能感兴趣的