Итерация по строкам в массиве Python для извлечения данных столбца - PullRequest
0 голосов
/ 11 февраля 2020

Я использую Python и пытаюсь перебрать каждую строку массива Nx9 и извлечь определенные значения из строки, чтобы сформировать из них другую матрицу. Значение N может меняться в зависимости от файла, который я читаю, но я использовал N = 3 в моем примере. Мне нужно только 0-е, 1-е, 3-е и 4-е значения каждой строки, чтобы сформировать в массив, который мне нужно сохранить. Например:

result = np.array([[1, 2, 3, 4, 5, 6, 7, 8, 9],
                  [11, 12, 13, 14, 15, 16, 17, 18, 19],
                  [21, 22, 23, 24, 25, 26, 27, 28, 29]])

#Output matrix of first row should be: ([[1,2],[4,5]])
#Output matrix of second row should be: ([[11,12],[14,15]])
#Output matrix of third row should be: ([[21,22],[24,25]])

Затем я должен получить N чисел, сформированных из извлеченных значений - двумерную матрицу для каждой строки. Однако сформированные матрицы выглядят трехмерными, поэтому при транспонировании и вычитании я получаю ошибку ValueError: operands could not be broadcast together with shapes (2,2,3) (3,2,2). Я знаю, что (3,2,2) матрица не может быть вычтена из (2,2,3), так как я могу получить 2D матрицу N раз? Al oop подойдет лучше? Есть предложения?

import numpy as np

result = np.array([[1, 2, 3, 4, 5, 6, 7, 8, 9],
                   [11, 12, 13, 14, 15, 16, 17, 18, 19],
                   [21, 22, 23, 24, 25, 26, 27, 28, 29]])

a = result[:, 0]
b = result[:, 1]
c = result[:, 2]
d = result[:, 3]
e = result[:, 4]
f = result[:, 5]
g = result[:, 6]
h = result[:, 7]
i = result[:, 8]
output = [[a, b], [d, e]]
output = np.array(output)
output_transpose = output.transpose()
result = 0.5 * (output - output_transpose)

Ответы [ 2 ]

0 голосов
/ 11 февраля 2020
In [276]: result = np.array( 
     ...:     [ 
     ...:         [1, 2, 3, 4, 5, 6, 7, 8, 9], 
     ...:         [11, 12, 13, 14, 15, 16, 17, 18, 19], 
     ...:         [21, 22, 23, 24, 25, 26, 27, 28, 29], 
     ...:     ] 
     ...: ) 
     ...:  
     ...: a = result[:, 0] 
          ...
     ...: i = result[:, 8] 
     ...: output = [[a, b], [d, e]]                                                            
In [277]: output                                                                               
Out[277]: 
[[array([ 1, 11, 21]), array([ 2, 12, 22])],
 [array([ 4, 14, 24]), array([ 5, 15, 25])]]
In [278]: arr = np.array(output)                                                               
In [279]: arr                                                                                  
Out[279]: 
array([[[ 1, 11, 21],
        [ 2, 12, 22]],

       [[ 4, 14, 24],
        [ 5, 15, 25]]])
In [280]: arr.shape                                                                            
Out[280]: (2, 2, 3)
In [281]: arr.T.shape                                                                          
Out[281]: (3, 2, 2)

transpose заменяет первое и последнее измерения.

Более чистый способ создания массива (N, 2,2) из ​​выбранных столбцов:

In [282]: arr = result[:,[0,1,3,4]].reshape(3,2,2)                                             
In [283]: arr.shape                                                                            
Out[283]: (3, 2, 2)
In [284]: arr                                                                                  
Out[284]: 
array([[[ 1,  2],
        [ 4,  5]],

       [[11, 12],
        [14, 15]],

       [[21, 22],
        [24, 25]]])

Поскольку последние 2 измерения равны 2, вы можете переставить их и взять разницу:

In [285]: arr-arr.transpose(0,2,1)                                                             
Out[285]: 
array([[[ 0, -2],
        [ 2,  0]],

       [[ 0, -2],
        [ 2,  0]],

       [[ 0, -2],
        [ 2,  0]]])

Другой способ получить массив (N, 2,2) - это индекс матрицы:

In [286]: result[:,[[0,1],[3,4]]]                                                              
Out[286]: 
array([[[ 1,  2],
        [ 4,  5]],

       [[11, 12],
        [14, 15]],

       [[21, 22],
        [24, 25]]])
0 голосов
/ 11 февраля 2020

Хорошо, это не проблема кодирования, а математическая проблема. Я написал некоторый код для вас, так как совершенно очевидно, что вы новичок, поэтому там будет какой-то незнакомый синтаксис, который вы должны изучить, чтобы избежать подобных проблем в будущем. Вы можете использовать их не так часто, но полезно знать, как это сделать, потому что это расширяет ваше понимание синтаксиса python в целом.

Прежде всего, полный код для простого копирования и вставки:

import numpy as np

result=np.array([[1,2,3,4,5,6,7,8,9],
                 [11,12,13,14,15,16,17,18,19],
                 [21,22,23,24,25,26,27,28,29]])

output = np.array(tuple(result[:,i] for i in (0,1,3)))


def Matrix_Operation(Matrix,Coefficient):

    if (Matrix.shape == Matrix.shape[::-1] 
        and isinstance(Matrix,np.ndarray)
        and isinstance(Coefficient,float)):

        return Coefficient*(Matrix-Matrix.transpose())
    else:
        print('The shape of you Matrix is not palindromic')
        print('You cannot substitute matrices of unequal shape')
        print('Your shape: %s'%str(Matrix.shape))

print(Matrix_Operation(output,0.5))

Теперь давайте поговорим о пошаговом объяснении того, что здесь происходит:

import numpy as np

result=np.array([[1,2,3,4,5,6,7,8,9],
                 [11,12,13,14,15,16,17,18,19],
                 [21,22,23,24,25,26,27,28,29]])

Python использует отступ (выравнивание пробелов) как неотъемлемую часть своего синтаксиса. Однако, если вы предоставляете скобки, большую часть времени вам не нужно выравнивать отступы, чтобы интерпретатор мог понять ваш код. Если вы предоставляете большой массив значений вручную, обычно рекомендуется начинать новые строки с запятых (здесь запятые, разделяющие подсписки). Это просто более читабельно, и, таким образом, ваши данные не будут отображаться за кадром в вашей программе кодирования.

output = np.array(tuple(result[:,i] for i in (0,1,3)))

Понимание списков является большой проблемой в python и действительно удобно для грязных строк. Насколько я знаю, никакой другой язык не дает вам такой возможности. Это одна из причин, почему python так здорово. Я в основном создал список списков, где каждый подсписок является результатом [:, i] для каждого i в (0,1,3). Это приведено как кортеж (да, постижение списков также может быть сделано с кортежами, а не только со списками). Наконец, я обернул его в функцию np.array, так как этот тип необходим для наших математических операций позже.

def Matrix_Operation(Matrix,Coefficient):

    if (Matrix.shape == Matrix.shape[::-1] 
        and isinstance(Matrix,np.ndarray)
        and isinstance(Coefficient,(float,int))):

        return Coefficient*(Matrix-Matrix.transpose())
    else:
        print('The shape of you Matrix is not palindromic')
        print('You cannot substitute matrices of unequal shape')
        print('Your shape: %s'%str(Matrix.shape))

print(Matrix_Operation(output,0.5))

Если вы собираетесь создать сложную формулу в python коде, почему бы и нет обернуть его внутри абстрактной функции? Вы также можете включить много «контроля качества» в функцию, чтобы проверить, правильно ли задан ввод для задачи, которую он должен выполнять.

Ваш код не выполнен, потому что вы пытались вычесть (2,2,3) -образная матрица из (3,2,2) -матрицы. Поэтому нам понадобится фрагмент кода, чтобы проверить, имеет ли наша предоставленная матрица форму палиндроми c. Вы можете изменить порядок элементов в контейнере, выполнив Container[::-1], и поэтому мы спрашиваем, если Matrix.shape == Matrix.shape [:: - 1]. Далее, необходимо, чтобы наша Матрица была np.ndarray и если наш коэффициент был числом. Вот что я делаю с функцией isinstance(). Вы можете проверить несколько типов одновременно, поэтому isinstance(Coefficient,(float,int)) содержит кортеж с int и float.

Теперь, когда мы убедились, что наш ввод имеет смысл, мы можем предварительно преобразовать нашу Matrix_Operation. Итак, в заключение: проверьте, если ваша математика solid, прежде чем просить SO о помощи, потому что люди здесь могут быть довольно раздражены на такие вещи. Вы, наверное, уже заметили, что кто-то уже отклонил ваш вопрос. Лично я считаю, что необходимо дать новичкам задавать пару глупых вопросов, прежде чем они попадут в паз, но для этого, как мне кажется, и есть кнопка голосования.

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