Search code examples
pythonnumpymatrix-multiplication

Weighted sum of multiple matrices


I have a list of m n x n matrices and a list of m real values (alphas). The values of n and m can be quite large. I am trying to calculate the weighted sum of the matrices with weights in alphas.

I was wondering if there is a numpy function (or any other library) that can do this faster than the manual for-loop method.

I have included my current function below.

def calculate_matrix_sums(mats, alphas):
    """
        Calculate the weighted sum of matrices in mats with weights alpha
    """
    k_mults = [np.multiply(mats[i], alphas[i]) for i in range(len(alphas))]
    k_sums1 = np.matrix(k_mults[0]) + np.matrix(k_mults[1])
    for i in range(2, len(k_mults)):
        k_sums1 = k_sums1 + np.asmatrix(k_mults[i])
    k_sums2 = np.asarray(k_sums1).astype(float)
    k_sums2 = k_sums2.reshape(len(mats[0]), len(mats[0]))
    return k_sums2

and a sample code:

matrices = np.asarray([np.array([[1., 0.77841638, 0.53239253, 0.9444068, 0.93024477],
                                 [0.77841638, 1., 0.7221497, 0.5805838, 0.68501944],
                                 [0.53239253, 0.7221497, 1., 0.36986265, 0.62792847],
                                 [0.9444068, 0.5805838, 0.36986265, 1., 0.88303226],
                                 [0.93024477, 0.68501944, 0.62792847, 0.88303226, 1.]]),
                       np.array([[1., 0.45650032, 0.13898701, 0.83605729, 0.79743304],
                                 [0.45650032, 1., 0.36094014, 0.18229867, 0.30596445],
                                 [0.13898701, 0.36094014, 1., 0.04443844, 0.23300302],
                                 [0.83605729, 0.18229867, 0.04443844, 1., 0.67745532],
                                 [0.79743304, 0.30596445, 0.23300302, 0.67745532, 1.]])])
alpha_vals = [0.47547796, 0.52452204]

print(calculate_matrix_sums(matrices, alpha_vals))

Any suggestions are appreciated.


Solution

  • You can reshape alpha_vals so that it broadcasts across the first axis of matrices correctly:

    (np.array(alpha_vals)[:, None, None] * matrices).sum(axis=0)
    

    Alternatively, you can adjust the strides of matrices, so that the last dimension corresponds to alpha_vals:

    (np.moveaxis(matrices, 0, -1) * alpha_vals).sum(axis=-1)
    

    You can also use np.einsum for this sort of thing (probably the most elegant solution):

    np.einsum('ijk,i->jk', matrices, alpha_vals)