задача вычисления обратной матрицы - PullRequest
0 голосов
/ 03 июля 2019

Я пытаюсь вычислить обратную квадратную матрицу любого ранга N x N. Я использую структуру для хранения значений матрицы, которые я могу эффективно и я уже в состоянии вычислить определитель.Но должна быть некоторая проблема с обратной функцией.Это код

    struct m{
        size_t row;
        size_t col;
        double *data;
    };

    void inverse(size_t n, struct m *A) /*Calculate the inverse of A */
{
    size_t i,j,i_count,j_count, count=0;
    double det = determinant(n, A);
    size_t id = 0;    
    double *d;

    struct m C; /*The Adjoint matrix */
    C.data = malloc(sizeof(double) * n * n);

    C.row = n;
    C.col = n;    

    struct m *minor; /*matrices obtained by removing the i row and j column*/

    if (!(minor = malloc(n*n*(n+1)*sizeof *minor))) {
        perror ("malloc-minor");
        exit(-1);
    }    

    if (det == 0){
        printf("The matrix is singular\n");
        exit(1);
    }    

    for(id=0; id < n*n; id++){
        d = minor[id].data = malloc(sizeof(double) * (n-1) * (n-1));
        for(count=0; count < n; count++)
        {
            //Creating array of Minors
            i_count = 0;
            for(i = 0; i < n; i++)
            {
                j_count=0;
                for(j = 0; j < n; j++)
                {
                    if(j == count)
                        continue; // don't copy the minor column element
                    *d = A->data[i * A->col + j];
                    d++;
                    j_count++;
                }
                i_count++;
            }
        }
    }

    for(id=0; id < n*n; id++){
        for(i=0; i < n; i++){
           for(j=0; j < n; j++) 
            C.data[i * C.col + j] = determinant(n-1,&minor[id]);//Recursive call
        }
    }

    transpose(&C);
    scalar_product(1/det, &C); 
    *A = C;    
}

Определитель вычисляется рекурсивно с помощью этого алгоритма:

double determinant(size_t n, struct m *A)
    {                                    
        size_t i,j,i_count,j_count, count=0;

        double det = 0;

        if(n < 1)
        {
            printf("Error\n");
            exit(1);
        }

        if(n==1) return A->data[0];

        else if(n==2) return (A->data[0]* A->data[1 * A->col + 1] - A->data[0 + 1] * A->data[1*A->col + 0]);    

        else{
            struct m C;

            C.row = A->row-1;
            C.col = A->col-1;

            C.data = malloc(sizeof(double) * (A->row-1) * (A->col-1));

            for(count=0; count < n; count++)
            {
                //Creating array of Minors
                i_count = 0;
                for(i = 1; i < n; i++)
                {
                    j_count=0;
                    for(j = 0; j < n; j++)
                    {
                        if(j == count)
                            continue; // don't copy the minor column element
                        C.data[i_count * C.col + j_count] = A->data[i * A->col + j];
                        j_count++;
                    }
                    i_count++;
                }
                det += pow(-1, count) * A->data[count] * determinant(n-1,&C);//Recursive call
            }
            free(C.data);
            return det;
        }
    }

Полный код можно найти здесь: https://ideone.com/gQRwVu.

Ответы [ 2 ]

0 голосов
/ 05 июля 2019

Ваше вычисление обратного не совсем соответствует алгоритму, описанному e. г. для Инверсия матрицы используя Minors, Cofactors и Adjugate , даже принимая во внимание, что вы на данный момент пропустили этап адъюгата и деления. Сравните ваш самый внешний цикл for в inverse() с этой рабочей реализацией:

double Rdata[(n-1)*(n-1)];              // remaining data values
struct m R = { n-1, n-1, Rdata };       // matrix structure for them
for (count = 0; count < n*n; count++)   // Create n*n Matrix of Minors
{
    int row = count/n, col = count%n;
    for (i_count = i = 0; i < n; i++)
        if (i != row)                   // don't copy the current row
        {
            for (j_count = j = 0; j < n; j++)
                if (j != col)           // don't copy the current column
                    Rdata[i_count*R.col+j_count++] = A->data[i*A->col+j];
            i_count++;
        }
    // transpose by swapping row and column
    C.data[col*C.col+row] = pow(-1, row&1 ^ col&1) * determinant(n-1, &R) / det;
}

Это дает для заданных входных данных правильную обратную матрицу

1  2 -4.5
0 -1  1.5
0  0  0.5

(уже транспонирован и разделен на определитель исходной матрицы).

Малые ноты:

  • *A = C; в конце inverse() теряет исходный указатель данных *A.
  • Функция форматирования f() неверна для отрицательных значений, поскольку дробь также отрицательна в этом случае. Вы могли бы написать if (fabs(f)<.00001).
0 голосов
/ 05 июля 2019

Используйте некоторую другую переменную в цикле после:

det + =pow(-1,count) * A->data[count] *determinant (n-1,&C)
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...