Оценка определителя с использованием оболочки LAPACK для декомпозиции LU в Cython - PullRequest
0 голосов
/ 11 мая 2018

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

from scipy.linalg.cython_lapack cimport dgetrf

cpdef double det_c(double[:, ::1] A, double[:, ::1] work, double[::1] ipiv):
    '''obtain determinant of float type square matrix A

    Notes
    -----
    As is, this function is not yet computing the sign of the determinant
    correctly, help!

    Parameters
    ----------
    A : memoryview (numpy array)
        n x n array to compute determinant of
    work : memoryview (numpy array)
        n x n array to use within function
    ipiv : memoryview (numpy array)
        length n vector use within function

    Returns
    -------
    detval : float
        determinant of matrix A
    '''

    cdef int n = A.shape[0], info
    work[...] = A

    dgetrf(&n, &n, &work[0,0], &n, &ipiv[0], &info)

    cdef double detval = 1.
    cdef int j

    for j in range(n):
        if j != ipiv[j]:
            detval = -detval*work[j, j]
        else:
            detval = detval*work[j, j]

    return detval

Когда я проверяю эту функцию и сравниваю ее с np.linalg.det, иногда я получаю неправильный знак.

>>> a = np.array([[1,2],[3,5.]])
>>> np.linalg.det(a)
>>> -1.0000000000000004
>>> det_c(a, np.zeros((2, 2)), np.zeros(2, dtype=np.int32))
>>> 1

В других случаях правильный знак.

>>> b = np.array([[1,2,3],[1,2,1],[5,6,1.]])
>>> np.linalg.det(b)
>>> -7.999999999999998
>>> det_c(a, np.zeros((3, 3)), np.zeros(3, dtype=np.int32))
>>> -8.0

1 Ответ

0 голосов
/ 11 мая 2018

dgetrf - это подпрограмма на Фортране, и Фортран использует индексирование на основе 1, поэтому значения в ipiv находятся в диапазоне от 1 до n (включительно).Чтобы учесть это, измените тест в цикле с

        if j != ipiv[j]:

на

        if j != ipiv[j] - 1:
...