Итерация по массиву

135

Есть ли менее многословная альтернатива этому:

for x in xrange(array.shape[0]):
    for y in xrange(array.shape[1]):
        do_stuff(x, y)

Я придумал это:

for x, y in itertools.product(map(xrange, array.shape)):
    do_stuff(x, y)

Который сохраняет один отступ, но все еще довольно уродлив.

Я надеюсь на что-то похожее на этот псевдокод:

for x, y in array.indices:
    do_stuff(x, y)

Существует ли что-нибудь подобное?

Рам Рахум
источник
Я в Python 2.7 и использую ваше решение с itertools; Я прочитал в комментариях, что с помощью itertools будет быстрее. однако (возможно, потому что я в 2.7) я также должен был распаковать карту в цикле for. for x, y in itertools.product(*map(xrange, array.shape)):
ALM
В справочнике NumPy есть страница под названием « Итерация по массивам
Кейси,
связанные: stackoverflow.com/questions/29493183/…
Eulenfuchswiesel

Ответы:

187

Я думаю, что вы ищете число .

>>> a =numpy.array([[1,2],[3,4],[5,6]])
>>> for (x,y), value in numpy.ndenumerate(a):
...  print x,y
... 
0 0
0 1
1 0
1 1
2 0
2 1

По поводу производительности. Это немного медленнее, чем понимание списка.

X = np.zeros((100, 100, 100))

%timeit list([((i,j,k), X[i,j,k]) for i in range(X.shape[0]) for j in range(X.shape[1]) for k in range(X.shape[2])])
1 loop, best of 3: 376 ms per loop

%timeit list(np.ndenumerate(X))
1 loop, best of 3: 570 ms per loop

Если вас беспокоит производительность, вы могли бы немного оптимизировать ее, взглянув на реализацию ndenumerate, которая выполняет две вещи: преобразование в массив и циклическое выполнение. Если вы знаете, что у вас есть массив, вы можете вызвать .coordsатрибут плоского итератора.

a = X.flat
%timeit list([(a.coords, x) for x in a.flat])
1 loop, best of 3: 305 ms per loop
SiggyF
источник
1
Обратите внимание, что это работает, но невероятно медленно. Тебе лучше повторять вручную.
Марти
43

Если вам нужны только индексы, вы можете попробовать numpy.ndindex:

>>> a = numpy.arange(9).reshape(3, 3)
>>> [(x, y) for x, y in numpy.ndindex(a.shape)]
[(0, 0), (0, 1), (0, 2), (1, 0), (1, 1), (1, 2), (2, 0), (2, 1), (2, 2)]
senderle
источник
15

увидеть nditer

import numpy as np
Y = np.array([3,4,5,6])
for y in np.nditer(Y, op_flags=['readwrite']):
    y += 3

Y == np.array([6, 7, 8, 9])

y = 3не будет работать, используйте y *= 0и y += 3вместо этого.

C19
источник
2
или используйте y [...] = 3
Дональд Хобсон