41

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

>>> 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

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

4

2 回答 2

51

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

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

类的主体(与类中定义的方法的主体相反)在创建类对象或绑定其名称之前(特别是在任何基础定义为之前class)作为语句的一部分执行作为基础——尽管这个最新的细节在提及裸名时永远不会重要!-)。

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

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

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

我,我喜欢barename 查找规则的简单性和通用性,并且我喜欢在任何时候我想要任何其他形式的查找时使用限定名称而不是barenames ......但是,这不是我疯狂爱上的秘密Python(我有自己的抱怨——例如,global x作为一个语句总是让我的皮肤爬行,我宁愿写在这里global.x,即,有global一个“当前正在执行的模块”的内置名称......我喜欢限定名称!-),是吗?-)

于 2010-09-06T02:28:09.340 回答
31

在 Python 中,类的主体在创建类之前在其自己的命名空间中执行(之后,该命名空间的成员成为该类的成员)。因此,当解释器到达 y = x+1 时,此时 B 类还不存在,因此没有父类。

有关更多详细信息,请参阅http://docs.python.org/reference/compound_stmts.html#class-definitions

于 2010-09-06T02:09:07.450 回答