From bd72a5d7f37af808a05b84fc37522d313fdf611f Mon Sep 17 00:00:00 2001 From: "Pavel N. Krivitsky" Date: Mon, 18 Dec 2023 18:22:03 +1100 Subject: [PATCH] Implemented ginv_eigen(), an alternative to MASS:ginv() that uses eigendecomposition instead of QR decompsition and its scaled and xTAx variants. Bumped minor version to 4.10 and imported Matrix. --- DESCRIPTION | 9 +++--- NAMESPACE | 3 ++ R/matrix.utils.R | 76 ++++++++++++++++++++++++++++++++++++++++-------- man/ssolve.Rd | 29 +++++++++++++----- man/xTAx.Rd | 16 ++++++++++ 5 files changed, 108 insertions(+), 25 deletions(-) diff --git a/DESCRIPTION b/DESCRIPTION index 8b9da83..3bfe8da 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -1,6 +1,6 @@ Package: statnet.common -Version: 4.9.0-436 -Date: 2023-06-27 +Version: 4.10.0-438 +Date: 2023-12-18 Title: Common R Scripts and Utilities Used by the Statnet Project Software Authors@R: c( person(c("Pavel", "N."), "Krivitsky", role=c("aut","cre"), email="pavel@statnet.org", comment=c(ORCID="0000-0002-9101-3362", affiliation="University of New South Wales")), @@ -8,7 +8,7 @@ Authors@R: c( person("Chad", "Klumb", role=c("ctb"), email="cklumb@gmail.com", comment=c(affiliation="University of Washington"))) Description: Non-statistical utilities used by the software developed by the Statnet Project. They may also be of use to others. Depends: R (>= 3.5) -Imports: utils, methods, coda, parallel, tools +Imports: utils, methods, coda, parallel, tools, Matrix BugReports: https://github.com/statnet/statnet.common/issues License: GPL-3 + file LICENSE URL: https://statnet.org @@ -17,5 +17,4 @@ RoxygenNote: 7.2.3 Encoding: UTF-8 Suggests: covr, rlang (>= 1.1.1), - MASS, - Matrix + MASS diff --git a/NAMESPACE b/NAMESPACE index ce89c57..ffb7833 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -86,6 +86,7 @@ export(eval_lhs.formula) export(filter_rhs.formula) export(fixed.pval) export(forkTimeout) +export(ginv_eigen) export(handle.controls) export(is.SPD) export(is.linwmatrix) @@ -140,8 +141,10 @@ export(update_snctrl) export(vector.namesmatch) export(xAxT) export(xTAx) +export(xTAx_eigen) export(xTAx_qrsolve) export(xTAx_qrssolve) +export(xTAx_seigen) export(xTAx_solve) export(xTAx_ssolve) importFrom(coda,as.mcmc) diff --git a/R/matrix.utils.R b/R/matrix.utils.R index f02ae27..9ef381c 100644 --- a/R/matrix.utils.R +++ b/R/matrix.utils.R @@ -85,6 +85,27 @@ sandwich_solve <- function(A, B, ...) { solve(A, t(solve(A, B, ...)), ...) } +#' @describeIn xTAx Evaluate \eqn{x' A^{-1} x} for vector \eqn{x} and +#' matrix \eqn{A} (symmetric, nonnegative-definite) via +#' eigendecomposition; returns `rank` and `nullity` as attributes +#' just in case subsequent calculations (e.g., hypothesis test +#' degrees of freedom) are affected. +#' +#' Decompose \eqn{A = P L P'} for \eqn{L} diagonal matrix of +#' eigenvalues and \eqn{P} orthogonal. Then \eqn{A^{-1} = P L^{-1} +#' P'}. +#' +#' Substituting, \deqn{x' A^{-1} x = x' P L^{-1} P' x +#' = h' L^{-1} h} for \eqn{h = P' x}. +#' +#' @export +xTAx_eigen <- function(x, A, tol=sqrt(.Machine$double.eps), ...) { + e <- eigen(A, symmetric=TRUE) + keep <- e$values > max(tol * e$values[1L], 0) + h <- drop(crossprod(x, e$vectors[, keep, drop=FALSE])) + structure(sum(h*h/e$values[keep]), rank = sum(keep), nullity = sum(!keep)) +} + #' Wrappers around matrix algebra functions that pre-*s*cale their #' arguments @@ -96,16 +117,23 @@ sandwich_solve <- function(A, B, ...) { #' functions first scale the matrix's rows and/or columns by its #' diagonal elements and then undo the scaling on the result. #' -#' `ssolve()`, `sginv()`, and `snearPD()` wrap [solve()], -#' [MASS::ginv()], and [Matrix::nearPD()], respectively. `srcond()` -#' returns the reciprocal condition number of [rcond()] net of the -#' above scaling. `xTAx_ssolve`, `xTAx_qrssolve`, and -#' `sandwich_ssolve` wrap the corresponding \pkg{statnet.common} -#' functions. +#' `ginv_eigen()` reimplements [MASS::ginv()] but using +#' eigendecomposition rather than SVD; this means that it is only +#' suitable for symmetric matrices, but that detection of negative +#' eigenvalues is more robust. +#' +#' `ssolve()`, `sginv()`, `sginv_eigen()`, and `snearPD()` wrap +#' [solve()], [MASS::ginv()], `ginv_eigen()`, and [Matrix::nearPD()], +#' respectively. `srcond()` returns the reciprocal condition number of +#' [rcond()] net of the above scaling. `xTAx_ssolve()`, +#' `xTAx_qrssolve()`, `xTAx_seigen()`, and `sandwich_ssolve()` wrap +#' the corresponding \pkg{statnet.common} functions. #' #' @param snnd assume that the matrix is symmetric non-negative -#' definite (SNND). If it's "obvious" that it's not (e.g., negative -#' diagonal elements), an error is raised. +#' definite (SNND). This typically entails scaling that converts +#' covariance to correlation and use of eigendecomposition rather +#' than singular-value decomposition. If it's "obvious" that the matrix is +#' not SSND (e.g., negative diagonal elements), an error is raised. #' #' @param x,a,b,X,A,B,tol,... corresponding arguments of the wrapped functions. #' @@ -135,7 +163,7 @@ ssolve <- function(a, b, ..., snnd = TRUE) { #' @rdname ssolve #' #' @export -sginv <- function(X, ..., snnd = TRUE) { +sginv <- function(X, ..., snnd = TRUE){ d <- diag(as.matrix(X)) d <- ifelse(d==0, 1, 1/d) @@ -143,14 +171,38 @@ sginv <- function(X, ..., snnd = TRUE) { d <- sqrt(d) if(anyNA(d)) stop("Matrix a has negative elements on the diagonal, and snnd=TRUE.") dd <- rep(d, each = length(d)) * d - X <- X * dd - MASS::ginv(X, ...) * dd + ginv_eigen(X * dd, ...) * dd } else { dd <- rep(d, each = length(d)) - MASS::ginv(X*d, ...) * dd + MASS::ginv(X * dd, ...) * t(dd) } } +#' @rdname ssolve +#' @export +ginv_eigen <- function(X, tol = sqrt(.Machine$double.eps), ...){ + e <- eigen(X, symmetric=TRUE) + keep <- e$values > max(tol * e$values[1L], 0) + tcrossprod(e$vectors[, keep, drop=FALSE] / rep(e$values[keep],each=ncol(X)), e$vectors[, keep, drop=FALSE]) +} + + +#' @rdname ssolve +#' +#' @export +xTAx_seigen <- function(x, A, tol=sqrt(.Machine$double.eps), ...) { + d <- diag(as.matrix(A)) + d <- ifelse(d<=0, 0, 1/d) + + d <- sqrt(d) + dd <- rep(d, each = length(d)) * d + + A <- A * dd + x <- x * d + + xTAx_eigen(x, A, tol=tol, ...) +} + #' @rdname ssolve #' #' @export diff --git a/man/ssolve.Rd b/man/ssolve.Rd index 2bb478b..b14142f 100644 --- a/man/ssolve.Rd +++ b/man/ssolve.Rd @@ -3,6 +3,8 @@ \name{ssolve} \alias{ssolve} \alias{sginv} +\alias{ginv_eigen} +\alias{xTAx_seigen} \alias{srcond} \alias{snearPD} \alias{xTAx_ssolve} @@ -15,6 +17,10 @@ ssolve(a, b, ..., snnd = TRUE) sginv(X, ..., snnd = TRUE) +ginv_eigen(X, tol = sqrt(.Machine$double.eps), ...) + +xTAx_seigen(x, A, tol = sqrt(.Machine$double.eps), ...) + srcond(x, ..., snnd = TRUE) snearPD(x, ...) @@ -27,8 +33,10 @@ sandwich_ssolve(A, B, ...) } \arguments{ \item{snnd}{assume that the matrix is symmetric non-negative -definite (SNND). If it's "obvious" that it's not (e.g., negative -diagonal elements), an error is raised.} +definite (SNND). This typically entails scaling that converts +covariance to correlation and use of eigendecomposition rather +than singular-value decomposition. If it's "obvious" that the matrix is +not SSND (e.g., negative diagonal elements), an error is raised.} \item{x, a, b, X, A, B, tol, ...}{corresponding arguments of the wrapped functions.} } @@ -41,12 +49,17 @@ functions first scale the matrix's rows and/or columns by its diagonal elements and then undo the scaling on the result. } \details{ -\code{ssolve()}, \code{sginv()}, and \code{snearPD()} wrap \code{\link[=solve]{solve()}}, -\code{\link[MASS:ginv]{MASS::ginv()}}, and \code{\link[Matrix:nearPD]{Matrix::nearPD()}}, respectively. \code{srcond()} -returns the reciprocal condition number of \code{\link[=rcond]{rcond()}} net of the -above scaling. \code{xTAx_ssolve}, \code{xTAx_qrssolve}, and -\code{sandwich_ssolve} wrap the corresponding \pkg{statnet.common} -functions. +\code{ginv_eigen()} reimplements \code{\link[MASS:ginv]{MASS::ginv()}} but using +eigendecomposition rather than SVD; this means that it is only +suitable for symmetric matrices, but that detection of negative +eigenvalues is more robust. + +\code{ssolve()}, \code{sginv()}, \code{sginv_eigen()}, and \code{snearPD()} wrap +\code{\link[=solve]{solve()}}, \code{\link[MASS:ginv]{MASS::ginv()}}, \code{ginv_eigen()}, and \code{\link[Matrix:nearPD]{Matrix::nearPD()}}, +respectively. \code{srcond()} returns the reciprocal condition number of +\code{\link[=rcond]{rcond()}} net of the above scaling. \code{xTAx_ssolve()}, +\code{xTAx_qrssolve()}, \code{xTAx_seigen()}, and \code{sandwich_ssolve()} wrap +the corresponding \pkg{statnet.common} functions. } \examples{ x <- rnorm(2, sd=c(1,1e12)) diff --git a/man/xTAx.Rd b/man/xTAx.Rd index 4bb067e..ceffe12 100644 --- a/man/xTAx.Rd +++ b/man/xTAx.Rd @@ -6,6 +6,7 @@ \alias{xTAx_solve} \alias{xTAx_qrsolve} \alias{sandwich_solve} +\alias{xTAx_eigen} \title{Common quadratic forms} \usage{ xTAx(x, A) @@ -17,6 +18,8 @@ xTAx_solve(x, A, ...) xTAx_qrsolve(x, A, tol = 1e-07, ...) sandwich_solve(A, B, ...) + +xTAx_eigen(x, A, tol = sqrt(.Machine$double.eps), ...) } \arguments{ \item{x}{a vector} @@ -56,4 +59,17 @@ and \code{nullity} as attributes just in case subsequent calculations \item \code{sandwich_solve()}: Evaluate \eqn{A^{-1}B(A')^{-1}} for \eqn{B} a square matrix and \eqn{A} invertible. +\item \code{xTAx_eigen()}: Evaluate \eqn{x' A^{-1} x} for vector \eqn{x} and +matrix \eqn{A} (symmetric, nonnegative-definite) via +eigendecomposition; returns \code{rank} and \code{nullity} as attributes +just in case subsequent calculations (e.g., hypothesis test +degrees of freedom) are affected. + +Decompose \eqn{A = P L P'} for \eqn{L} diagonal matrix of +eigenvalues and \eqn{P} orthogonal. Then \eqn{A^{-1} = P L^{-1} + P'}. + +Substituting, \deqn{x' A^{-1} x = x' P L^{-1} P' x + = h' L^{-1} h} for \eqn{h = P' x}. + }}