Создание 4D вида на массиве 2D, чтобы разделить его на ячейки фиксированного размера - PullRequest
8 голосов
/ 25 января 2012

У меня есть двумерный массив t в numpy:

>>> t = numpy.array(range(81)).reshape((9,9))
>>> t
array([[ 0,  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]])

Он проиндексирован двумя числами: индекс строки и столбца.

>>> t[2,3]
21
>>> t.shape
(9, 9)
>>> t.strides
(72, 8)

Что я хочу сделать разделить массив на прямоугольные ячейки фиксированного размера , например, 3 × 3.Я хотел бы избежать копирования памяти.Я пытаюсь добиться этого путем создания вида на t с соответствующей формой и шагами ((3,3,3,3) и (216,24,72,8) соответственно).Таким образом, первые два индекса представления будут означать положение ячейки 3 × 3 в большей сетке, а последние два будут означать положение элемента внутри ячейки.Например, t[0,1,:,:] вернет

array([[ 3,  4,  5],
       [12, 13, 14],
       [21, 22, 23]])

Итак, мой вопрос - как создать описанное представление?Я пропускаю более простой метод?Можно ли это сделать элегантно с помощью синтаксиса срезов?

Ответы [ 2 ]

6 голосов
/ 25 января 2012

Редактировать : способ, который не требует от вас самостоятельного определения шагов,

numpy.rollaxis(t.reshape(3, 3, 3, 3), 2, 1)

[конец редактирования]

Еще один способ добиться этогодолжен использовать numpy.lib.stride_tricks.as_strided:

>>> t = numpy.arange(81.).reshape((9,9))
>>> numpy.lib.stride_tricks.as_strided(t, shape=(3,3,3,3), strides=(216,24,72,8))
array([[[[  0.,   1.,   2.],
         [  9.,  10.,  11.],
         [ 18.,  19.,  20.]],

        [[  3.,   4.,   5.],
         [ 12.,  13.,  14.],
         [ 21.,  22.,  23.]],

        [[  6.,   7.,   8.],
         [ 15.,  16.,  17.],
         [ 24.,  25.,  26.]]],


       [[[ 27.,  28.,  29.],
         [ 36.,  37.,  38.],
         [ 45.,  46.,  47.]],

        [[ 30.,  31.,  32.],
         [ 39.,  40.,  41.],
         [ 48.,  49.,  50.]],

        [[ 33.,  34.,  35.],
         [ 42.,  43.,  44.],
         [ 51.,  52.,  53.]]],


       [[[ 54.,  55.,  56.],
         [ 63.,  64.,  65.],
         [ 72.,  73.,  74.]],

        [[ 57.,  58.,  59.],
         [ 66.,  67.,  68.],
         [ 75.,  76.,  77.]],

        [[ 60.,  61.,  62.],
         [ 69.,  70.,  71.],
         [ 78.,  79.,  80.]]]])

Обратите внимание, что указанные вами шаги верны только для массивов с плавающей запятой (itemsize == 8), в то время как пример t в вашем посте является массивом int (которые могут иметь или не иметь itemsize == 8).

2 голосов
/ 25 января 2012

Вы можете сделать:

t = np.arange(81).reshape(9,9)
t.shape = (3, 3, 3, 3)
t = t.transpose((0, 2, 1, 3))

>>> print t.strides
(108, 12, 36, 4)

>>> print t
[[[[ 0  1  2]
   [ 9 10 11]
   [18 19 20]]

  [[ 3  4  5]
   [12 13 14]
   [21 22 23]]

  [[ 6  7  8]
   [15 16 17]
   [24 25 26]]]


 [[[27 28 29]
   [36 37 38]
   [45 46 47]]

  [[30 31 32]
   [39 40 41]
   [48 49 50]]

  [[33 34 35]
   [42 43 44]
   [51 52 53]]]


 [[[54 55 56]
   [63 64 65]
   [72 73 74]]

  [[57 58 59]
   [66 67 68]
   [75 76 77]]

  [[60 61 62]
   [69 70 71]
   [78 79 80]]]]

transpose вернет представление, когда это возможно, так что вам не нужно беспокоиться о знании типа данных.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...