Реализация псевдокода для обновления QR - какая часть моего кода неверна?

Я реализую некоторый псевдокод для обновления QR-разложения - я хочу удалить строку из матрицы A и обновить Q а также R соответственно. Мои результаты отличаются от результатов, полученных Scipy's qr_delete() хотя (см. строку 21 в прилагаемом фрагменте кода) ни Q, ни R не совпадают.

Псевдокод для обновления, на которое я ссылаюсь, можно найти здесь для алгоритма 2.1 и здесь для алгоритма 2.2.

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

import numpy as np
from typing import Tuple
import math
import scipy.linalg._decomp_update as scipy_qr_update


# Generate a random matrix A and compute QR decomposition.
m, n = (15, 10)
A = np.random.rand(m, n)
x = np.ones((n, 1))
b = np.dot(A, x)
Q, R = scipy.linalg.qr(A)

# Compute update with scipy's qr_delete().
Q_tilde_corr, R_tilde_corr = scipy_qr_update.qr_delete(Q, R, k=0, p=m - m_tilde, which="row")

# Compute update with own implementation.
Q_tilde, R_tilde, b_tilde, residual = qr_delete_row(np.copy(Q), np.copy(R), np.copy(b), k=0)

# Check if scipy's and our results are equal.
print(np.allclose(Q_tilde_corr, Q_tilde), np.allclose(R_tilde_corr, R_tilde))

def qr_delete_row(Q: np.ndarray, R: np.ndarray, b: np.ndarray, k: int) -> Tuple[np.ndarray, np.ndarray, np.ndarray, np.float]:
    """
    Updates Q and R after deleting one single row with index k.
    :param Q:
    :param R:
    :param b: b in Ax = b.
    :param k:
    :return:
    """

    m, n = Q.shape[0], R.shape[1]

    ###################################
    # Algorithm 2.1 - compute R_tilde.
    ###################################

    q_t = Q[k, :]
    q = q_t.T
    cs_values = np.zeros((m, 2))
    cs_values[0] = givens(q[0], q[1])

    if k != 0:
        b[1:k + 1] = b[0:k]

    d = Q.T @ b

    for j in np.arange(start=m - 2, step=-1, stop=-1):
        c, s = givens(q[j], q[j + 1])
        cs_values[j] = [c, s]
        cs_matrix = np.asarray([[c, s], [-s, c]])

        q[j] = c * q[j] - s * q[j + 1]

        if j <= n:
            R[j:j + 2, j:] = cs_matrix.T @ R[j:j + 2, j:]

        d[j: j + 2] = cs_matrix.T @ d[j:j + 2]

    R_tilde = R[1:, :]
    d_tilde = d[1:]
    resid = np.linalg.norm(d_tilde[n + 1:m], ord=2)

    ###################################
    # Algorithm 2.2 - compute Q_tilde.
    ###################################

    if k != 0:
        Q[1:k + 1, 1:] = Q[0:k, 0:]

    for j in np.arange(start=m - 2, step=-1, stop=-1):
        c, s = cs_values[j]
        cs_matrix = np.asarray([[c, s], [-s, c]])

        Q[1:, j:j + 2] = Q[1:m, j:j + 2] @ cs_matrix

    Q[1:, 1] = cs_values[0, 1] * Q[1:, 0] + cs_values[0, 1] * Q[1:, 1]

    return Q[1:, 1:], R_tilde, b[1:], resid

0 ответов

Другие вопросы по тегам