namedtuple 的第一个参数是做什么用的?

What's the first argument of namedtuple used for?

我们这样使用namedtuple

>>> from collections import namedtuple
>>> Point = namedtuple('Point', ['x', 'y'])
>>> p=Point(1,2)
>>> p.x
1

我发现 namedtuple 的第一个参数似乎没用,因为:

首先,我们不能使用它(例如创建一个实例):

>>> from collections import namedtuple
>>> P = namedtuple('Point', ['x', 'y'])
>>> p = Point(1,2)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
NameError: name 'Point' is not defined

其次,似乎没有限制(例如,我们不必使其唯一):

>>> P1 = namedtuple('Point', ['x', 'y'])
>>> P2 = namedtuple('Point', ['x', 'y', 'z'])
>>> p1 = P1(1,2)
>>> p2 = P2(1,2,3)
>>> p1
Point(x=1, y=2)
>>> p2
Point(x=1, y=2, z=3)

我没有从 manual or by googling. There is a relevant question here 中找到解释,但它没有回答为什么 namedtuple 需要第一个参数以及如何使用它或何时需要它。

设置生成的名称class:

>>> from collections import namedtuple
>>> Point = namedtuple('Point', ['x', 'y'])
>>> Point
<class '__main__.Point'>
>>> Point = namedtuple('Foobar', ['x', 'y'])
>>> Point
<class '__main__.Foobar'>

全局变量中的Point名称只是对生成的class的引用,class本身需要有一个名称.名字不能取自对象赋值的变量,因为class生成first,只有then赋值在一个单独的步骤(在 namedtuple() 调用 returns 之后)。

它不需要是唯一的,就像使用 class 时 class 名称不需要是唯一的一样:

>>> class Foo(object):
...     pass
... 
>>> Bar = Foo
>>> class Foo(object):
...     pass
... 
>>> Bar
<class '__main__.Foo'>
>>> Foo
<class '__main__.Foo'>

您通常通过使每个模块的名称唯一来避免这种情况,但您没有

该名称用于调试目的;在回溯和 repr() 输出中显示的名称是为了帮助您弄清楚 class 使用了什么。

对于Python,如果生成的class重用一个名字,没关系。它对 pickle 模块起作用;如果您需要能够 pickle 命名元组 class 实例,请确保您选择的名称与全局匹配,pickle 稍后可以通过该名称再次加载它。但是,如果重复使用某个名称对您来说有意义,或者不会积极阻碍您的代码开发或调试任务,那么请务必重复使用该名称或选择您喜欢的任何名称。