为什么 Python 二维数组索引顺序与冒号 (:) 一起使用时无关紧要

Why doesn't the Python 2D array index order matter when used with colon ( : )

创建二维数组,例如

x = [range(i, i+10) for i in xrange(1,100,10)]

并像这样使用冒号运算符进行索引

>>> x[2][:]
[21, 22, 23, 24, 25, 26, 27, 28, 29, 30]

按预期工作。它 returns 第 2 行的所有内容。

但是,如果我想检索所有第 2 列,我会本能地这样做

>>> x[:][2]

但这也returns

[21, 22, 23, 24, 25, 26, 27, 28, 29, 30]

这背后的原因是什么?我会凭直觉认为这 returns 是每行的第 2 列。

(此外,我知道我可以使用 numpy 来执行 x[:,2] 或者我可以使用列表理解来完成此操作,这不是我的问题)

x[2][:] 将 return 数组的第 3 行,然后 return 该行内的所有元素。

x[:][2] 将 return 数组的所有行,然后 return 所有行的第 3 行。

实际上,它们是相同的,x[2][:] == x[:][2]

在控制台中打印它们很明显为什么会这样:

In [138]: x[2]
Out[138]: [21, 22, 23, 24, 25, 26, 27, 28, 29, 30]

In [139]: x[2][:]
Out[139]: [21, 22, 23, 24, 25, 26, 27, 28, 29, 30]

In [140]: x[:]
Out[140]:
[[1, 2, 3, 4, 5, 6, 7, 8, 9, 10],
 [11, 12, 13, 14, 15, 16, 17, 18, 19, 20],
 [21, 22, 23, 24, 25, 26, 27, 28, 29, 30],
 [31, 32, 33, 34, 35, 36, 37, 38, 39, 40],
 [41, 42, 43, 44, 45, 46, 47, 48, 49, 50],
 [51, 52, 53, 54, 55, 56, 57, 58, 59, 60],
 [61, 62, 63, 64, 65, 66, 67, 68, 69, 70],
 [71, 72, 73, 74, 75, 76, 77, 78, 79, 80],
 [81, 82, 83, 84, 85, 86, 87, 88, 89, 90],
 [91, 92, 93, 94, 95, 96, 97, 98, 99, 100]]

In [141]: x[:][2]
Out[141]: [21, 22, 23, 24, 25, 26, 27, 28, 29, 30]

如果您希望获取列,那么您可以使用 list(zip(*x)) 有效地转置列表(在 Py 3 中添加了 list)。然后你可以这样做:

In [3]: list(zip(*x))[2]
Out[3]: (3, 13, 23, 33, 43, 53, 63, 73, 83, 93)

除此之外,我建议您使用 numpy 代替此类工作。

原因是[:]就是"everything",一行中的两个索引操作是完全独立的。

y = x[2][:]

tmp = x[2]
y = tmp[:]  # this only makes a copy, does nothing else

同样,

y = x[:][2]

tmp = x[:]  # this only makes a copy, does nothing else
y = tmp[2]

实际上两者都意味着

y = x[2]

任何时候都没有 2D 索引,Python 没有 2D 索引(尽管 numpy 有技巧使 像实际一样工作二维索引正在进行中)。

仅供以后查看此内容的任何人注意。正如其他人在上面提到的,arr[:][x] 将解析如下。

  • 临时数组将等于 arr[:] 这就是一切。
  • 那么你的最终数组将等于temp[x]

由于你的第一部分没有降低维度,第二个语句将被强制获取一行。

在您的反向版本中 arr[x][:] 完成以下操作:

  • 临时数组将等于 arr 的第 x 行
  • 然后您将 return 整个行,因为“:”

我知道你不想问关于 numpy 的事,而是想问其他想知道的人。 numpy 语法 arr[x,:]arr[:,x] WILL 也会按照您想要的方式工作。第一个将获取第 x 行中的所有值,第二个将获取第 x 列中的所有值。请注意,如果您正在使用列表,则必须先为其创建一个 np 数组。