python子类访问父类变量

标签 python subclass class-variables

我惊讶地发现,子类的类变量如果不明确指出父类的类名,就无法访​​问父类的类变量:

>>> class A(object):
...     x = 0
... 
>>> class B(A):
...     y = x+1
... 
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "<stdin>", line 2, in B
NameError: name 'x' is not defined
>>> class B(A):
...     y = A.x + 1
... 
>>> B.x
0
>>> B.y
1

为什么在定义 B.y 时我必须引用 A.x 而不仅仅是 x?这与我对实例变量的直觉相反,因为我可以在定义 B 之后引用 B.x。

最佳答案

Python 对裸名称的作用域规则非常简单明了:首先是本地 namespace ,然后(如果有的话)当前嵌套在其中的外部函数,然后是全局函数,最后是内置函数。这就是查找裸名时所发生的一切,并且不需要记住或应用任何复杂的规则(也不需要 Python 编译器来强制执行更复杂的规则)。

任何时候你想要一个不同的查找,你都将使用一个合格的名称,而不是一个简单的名称。限定名称的功能要强大得多,因为查找总是可以委托(delegate)给可以请求其属性的对象,并且这些对象可以实现他们需要的任何查找规则。特别是,在类中的实例方法中,self.x 要求 self 对象查找属性名称 的方法>'x' -- 并且在该查找中它可以委托(delegate)给类,包括继承概念的实现(以及多重继承、方法解析顺序等)。

类的主体(与类中定义的方法的主体相反)作为class语句的一部分执行,before 创建类对象或绑定(bind)它的名称(特别是在任何基被定义为基之前——尽管这个最新的细节在引用裸名时永远无关紧要!-)。

所以,在您的示例中,在类 B 中,barename x 是使用通用规则查找的——它是本地绑定(bind)的名称吗?如果不是,它是否绑定(bind)在嵌套此范围的任何外部函数中?如果不是,它是绑定(bind)为全局的还是内置的?如果以上都不是,使用有问题的裸名当然会导致名称错误异常。

由于您想要的查找顺序与普遍强制执行的裸名查找规则不同,那么显然您需要使用限定名称,而不是裸名;片刻的反射(reflection)将清楚地表明,用于您的目的的限定名称的“一个明显的选择”必须是 A.x - 因为这是您想要它的地方被查找(在那个时候,基地还没有被记录在任何地方,毕竟......它将是元类,通常是type,它将做基地- 绑定(bind)作为其工作的一部分,当它在类主体完成执行后被调用!-)。

有些人如此热衷于查找裸名的其他“神奇”规则,以至于他们无法忍受 Python 的这一方面(我相信最初是受到 Modula-3 的启发,Modula-3 是一种鲜为人知的非常好的语言在理论家的圈子中考虑过;-) -- 必须在方法中编写 self.x 来指定必须在 self 上查找 x例如,而不是使用通用的裸名规则,会让这些人发疯。

我,我喜欢准名称查找规则的简单性和通用性,并且我喜欢在任何时候使用限定名称而不是准名称来任何其他形式的查找...但是,它不是一个我疯狂地爱上 Python 的 secret (我有自己的提示——例如,global x 作为一个声明总是让我的皮肤毛骨悚然,我宁愿写 global .x,即让 global 成为“当前正在执行的模块”的内置名称...我确实喜欢限定名称!-),是吗?-)

关于python子类访问父类变量,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3648564/

相关文章:

java - 属于类变量还是不属于类变量?

delphi - "class var"是否初始化为零?

python - uint8 numpy数组的内存效率绝对差

python - 如何停止在 Tkinter 中引发事件?

python - 如何将 elasticsearch 库的日志记录级别设置为与我自己的日志记录不同?

java - ClassOne 中的 doSomething() 无法实现 InterfaceOne 中的 doSomething(),尝试分配较弱的访问权限,已公开

java继承处理字段

具有相同方法的 C# 子类

python - 使用 python 中的虚拟文件改进 CSV 推送到 MySQL

ruby-on-rails - 用于记录的 Rails 实例变量