python的“私有属性”实为命名改写机制:单下划线\_xxx是受保护约定,无语言限制;双下划线\_\_xxx触发\_classname\_\_name改写,防子类冲突但非真正私有,体现“我们都是有责任心的成年人”哲学。

Python 中的“私有属性”并不是真正意义上的私有,而是一种命名改写(name mangling)机制,目的是避免子类意外覆盖父类的内部属性,同时起到弱提示作用:这个属性不建议外部直接访问。
下划线约定:单下划线 _xxx 是“受保护”的提示
以单下划线开头的属性(如 _value)属于编程惯例,并不会触发任何语言级别的限制。它只是告诉其他开发者:“这个属性是内部使用的,请尽量不要直接访问”。解释器完全允许读写,IDE 可能会标为灰色或给出警告,但运行时毫无阻碍。
- 可以被子类继承和访问
- 可以被模块外代码直接调用(比如 obj._value)
- 不会被 from module import * 导入(仅限模块级变量)
双下划线 __xxx 触发命名改写(Name Mangling)
以双下划线开头、且不以双下划线结尾的名称(如 __value),会在类定义时被自动重命名为 _ClassName__value。这是 Python 解释器做的静态改名,发生在编译阶段,不是运行时动态行为。
- 改写只在类定义体内生效(方法中对 self.__x 的引用会被改写)
- 子类若定义同名 __x,会被改写为 _SubClass__x,与父类不冲突
- 仍可通过改写后的名字(如 obj._MyClass__value)强行访问——所以不是真正的私有
为什么不用真正的访问控制?
Python 坚持“我们都是 consenting adults”(我们都是有责任心的成年人)的设计哲学。语言不强制封装,而是靠约定、文档和工具(如 type checkers、linter)来辅助规范。过度封装反而增加维护成本,也违背 Python 的简洁与透明原则。
立即学习“Python免费学习笔记(深入)”;
- __slots__ 可限制实例属性,但不等于私有化
- @property + 私有字段可实现可控访问,但底层仍可绕过
- 真正敏感数据应靠逻辑隔离(如权限校验)、加密或外部服务,而非靠属性名隐藏
实用建议:怎么合理用好“伪私有”?
把双下划线当作“避免命名冲突的工具”,而不是“安全屏障”。日常开发中:
- 用 _xxx 表示“请勿直接使用”的内部属性或方法
- 仅在需要防止子类意外覆盖时,才用 __xxx(例如基类中关键的钩子字段)
- 对外暴露的接口一律用普通名称(value),配合 @property 控制读写逻辑
- 别依赖 __xxx 防止恶意访问——它挡不住有意为之的调用










