The sparseMVN package consists of two user-facing functions: dmvn.sparse() computes the density of a multivariate normal (MVN) distribution, and rmvn.sparse() samples from an MVN distribution. Unlike MVN functions provided by the mvtnorm and [MASS] packages, which take a dense representation of a full covariance matrix as one of the arguments, the sparseMVN functions ask for a sparse Cholesky decomposition of either the covariance or precision matrix. When the dimension of the MVN random variable is large, and the true covariance is itself sparse (i.e., the proportion of nonzero elements is small), sparseMVN offers the following advantages.

  1. Memory use and computation time in matrix operations: A dense matrix in R stores all M2 elements, even if the matrix is symmetric, and even when a small proportion of elements are nonzero. Thus, memory requirements grow quadratically with the size of the problem. Further, dense Cholesky factorization is a O(M3) operation, while multiplication of a triangular matrix with a dense matrix, and solving dense triangular systems, are O(M2) (Golub and Van Loan 1996). sparseMVN replies on classes from the Matrix package that compresses sparse matrices into an efficient structure for storage, and provides optimized algorithms for operations on those matrices. So, working with MVN distributions becomes more scalable when the covariance or precision matrix is actually sparse. The reasons are that only nonzero elements are stored explicitly, and redundant multiplications with zero are avoided.

  2. Avoiding explicit inversion of a precision matrix. In many applications (e.g., sampling from a Laplace approximation), the user starts with a sparse precision matrix rather than a covariance. Using extant MVN functions in R requires the user to explicitly invert the precision matrix, which itself can be an expensive step, with no guarantee of sparsity afterward. But this inversion step is not mathematically necessary; computation of MVN densities and random variates can be done just as easily and directly with the precision matrix. At the user’s option, sparseMVN accepts a factor of either a sparse covariance or precision matrix.

  3. Multiple-use Cholesky factorization. MVN computation involves computing the Cholesky factor of the matrix (either covariance or precision). Most MVN functions do this internally (so the user does not have to worry about it) every time the function is called (which is wasteful if the matrix does not change from call to call). sparseMVN requires the user to perform this step separately with Matrix::Cholesky(). While this feature does impose one additional step on the user, it also allows the user to reuse the same factor when possible.

Background

Let xRM be a realization of random variable XMVN(μ,Σ), where μRM is a vector, ΣRM×M is a positive-definite covariance matrix, and Σ1RM×M is a positive-definite precision matrix.

The log probability density of x is

logf(x)=12(Mlog(2π)+log|Σ|+zz),where zz=(xμ)Σ1(xμ)

MVN density computation and random number generation

The two computationally intensive steps in evaluating logf(x) are computing log|Σ|, and zz, without explicitly inverting Σ or repeating mathematical operations. But once we have z, computing the dot product zz is cheap. How to perform these steps efficiently in practice depends on whether the covariance matrix Σ, or the precision matrix Σ1 is available. For both cases, we start by finding a lower triangular matrix root: Σ=LL or Σ1=ΛΛ. Since Σ and Σ1 are positive definite, we will use the Cholesky decomposition, which is the unique matrix root with all positive elements on the diagonal.

With the Cholesky decomposition in hand, we compute the log determinant of Σ by adding the logs of the diagonal elements of the factors. log|Σ|={2Mm=1logLmm when Σ is given2Mm=1logΛmm when Σ1 is given

Having already computed the triangular matrix roots also speeds up the computation of zz. If Σ1 is given, z=Λ(xμ) can be computed efficiently as the product of an upper triangular matrix and a vector. When Σ is given, we find z by solving the lower triangular system Lz=xμ. The subsequent zz computation is trivially fast.

The algorithm for simulating XMVN(μ,Σ) also depends on whether Σ or Σ1 is given. As above, we start by computing the Cholesky decomposition of the given covariance or precision matrix. Define a random variable ZMVN(0,IM), and generate a realization z as a vector of M samples from a standard normal distribution. If Σ is given, then evaluate x=Lz+μ. If Σ1 is given, then solve for x in the triangular linear system Λ(xμ)=z. The resulting x is a sample from MVN(μ,Σ). The following shows that this approach is correct:

E(X)=E(LZ+μ)=E(ΛZ+μ)=μcov(X)=cov(LZ+μ)=E(LZZL)=LL=Σcov(X)=cov((Λ)1Z+μ)=E((Λ)1ZZΛ1)=(Λ)1Λ1=(ΛΛ)1=Σ

These algorithms apply when the covariance/precision matrix is either sparse or dense. When the matrix is dense, the computational complexity is O(M3) for a Cholesky decomposition, and O(M2) for either solving the triangular linear system or multiplying a triangular matrix by another matrix (Golub and Van Loan 1996). Thus, the computational cost grows cubically with M before the decomposition step, and quadratically if the decomposition has already been completed. Additionally, the storage requirement for Σ (or Σ1) grows quadratically with M.

Sparse matrices in R

The Matrix package (Bates and Maechler 2017) defines various classes for storing sparse matrices in compressed formats. The most important class for our purposes is dsCMatrix, which defines a symmetric matrix, with numeric (double precision) elements, in a column-compressed format. Three vectors define the underlying matrix: the unique nonzero values (just one triangle is needed), the indices in the value vector for the first value in each column, and the indices of the rows in which each value is located. The storage requirements for a sparse M×M symmetric matrix with V unique nonzero elements in one triangle are for V double precision numbers, V+M+1 integers, and some metadata. In contrast, a dense representation of the same matrix stores M2 double precision values, regardless of symmetry and the number of zeros. If V grows more slowly than M2, the matrix becomes increasingly sparse (a smaller percentage of elements are nonzero), with greater efficiency gains from storing the matrix in a compressed sparse format.

To illustrate how sparse matrices require less memory resources when compressed than when stored densely, consider the following example, which borrows heavily from the vignette of the sparseHessianFD package.

Suppose we have a dataset of N households, each with T opportunities to purchase a particular product. Let yi be the number of times household i purchases the product, out of the T purchase opportunities, and let pi be the probability of purchase. The heterogeneous parameter pi is the same for all T opportunities, so yi is a binomial random variable.

Let βiRk be a heterogeneous coefficient vector that is specific to household i, such that βi=(βi1,,βik). Similarly, wiRk is a vector of household-specific covariates. Define each pi such that the log odds of pi is a linear function of βi and wi, but does not depend directly on βj and wj for another household ji: pi=exp(wiβi)1+exp(wiβi), i=1,,N

The coefficient vectors βi are distributed across the population of households following a MVN distribution with mean μRk and covariance ARk×k. Assume that we know A, but not μ, so we place a multivariate normal prior on μ, with mean 0 and covariance ΩRk×k. Thus, the parameter vector xR(N+1)k consists of the Nk elements in the N βi vectors, and the k elements in μ.

The log posterior density, ignoring any normalization constants, is logπ(β1:N,μ|Y,W,A,Ω)=Ni=1(pyii(1pi)Tyi12(βiμ)A1(βiμ))12μΩ1μ

Because one element of βi can be correlated with another element of βi (for the same unit), we allow for the cross-partials between elements of βi for any i to be nonzero. Also, because the mean of each βi depends on μ, the cross-partials between μ and any βi can be nonzero. However, since the uildβi and βj are independent samples, and the yi are conditionally independent, the cross-partial derivatives between an element of βi and any element of any βj for ji, must be zero. When N is much greater than k, there will be many more zero cross-partial derivatives than nonzero, and the Hessian of the log posterior density will be sparse.

The sparsity pattern depends on how the variables are ordered. One such ordering is to group all of the coefficients in the βi for each unit together, and place μ at the end. β11,,β1k,β21,,β2k,  ,βN1,,βNk,μ1,,μk

In this case, the Hessian has a “block-arrow” pattern. This figure illustrates this pattern for N=5 and k=2 (12 total variables).

Block arrow (left) and banded (right) sparsity patternsBlock arrow (left) and banded (right) sparsity patterns

Block arrow (left) and banded (right) sparsity patterns

Another possibility is to group coefficients for each covariate together. β11,,βN1,β12,,βN2,  ,β1k,,βNk,μ1,,μk

Now the Hessian has an "banded" sparsity pattern:

In both cases, the number of nonzeros is the same. There are 144 elements in this symmetric matrix. If the matrix is stored in the standard base R dense format, memory is reserved for all 144 values, even though only 64 values are nonzero, and only 38 values are unique. For larger matrices, the reduction in memory requirements by storing the matrix in a sparse format can be substantial. If N=1,000 and k=2, then M= 2,002, with more than 4 million elements in the Hessian. However, only 12,004 of those elements are nonzero, with 7,003 unique values in the lower triangle. The dense matrix requires 30.6 Mb of RAM, while a sparse symmetric dsCMatrix matrix requires only 91.6 Kb.

This example is relevant because, when evaluated at the posterior mode, the Hessian matrix of the log posterior is the MVN precision matrix Σ1 of a MVN approximation to the posterior distribution of (β1:N,μ). If we were to simulate from this MVN using , or evaluate MVN densities using , we would need to invert Σ1 to Σ first, and store it as a dense matrix. Internally, and invoke dense linear algebra routines, including matrix factorization.

Using the sparseMVN package

The signatures of the key sparseMVN functions are

rmvn.sparse(n, mu, CH, prec=TRUE, log=TRUE)
dmvn.sparse(x, mu, CH, prec=TRUE, log=TRUE)

rmvn.sparse() returns a matrix x with n rows and length(mu) columns. dmvn.sparse() returns a vector of length n: densities if log=FALSE , and log densities if log=TRUE.

x A numeric matrix. Each row is an MVN sample.
mu A numeric vector. The mean of the MVN random variable.
CH Either a dCHMsimpl or dCHMsuper object representing the Cholesky decomposition of the covariance/precision matrix.
prec Logical value that identifies CH as the Cholesky decomposition of either a covariance (Σ, ) or precision(Σ1, ) matrix.
n Number of random samples to be generated.
log If log=TRUE, the log density is returned.

[Table 1: Arguments to dmvn.sparse() and rmvn.sparse()]

dmvn.sparse() and rmvn.sparse() require the user to compute the Cholesky decomposition CH beforehand, but this needs to be done only once (as long as Σ or Σ1 does not change). CH should be computed using Cholesky(), whose first argument is a sparse symmetric matrix stored as a dsCMatrix object. As far as we know, there is no particular need to deviate from the defaults of the remaining arguments. If Cholesky() uses a fill-reducing permutation to compute CH , the sparseMVN functions will handle that directly, with no additional user intervention required. The chol() function in base R should not be used.

An example

Suppose we want to generate samples from an MVN approximation to the posterior distribution of our example model from Section 1.2. sparseMVN includes functions to simulate data and compute the log posterior density, gradient and Hessian for the binary choice example model. The trustOptim package provides a nonlinear optimizer that estimates the curvature of the objective function using a sparse Hessian.

The calls to trust.optim() return the posterior mode, and the Hessian evaluated at the mode. These results serve as the mean and the negative precision of the MVN approximation to the posterior distribution of the model.

We can then sample from the posterior using an MVN approximation, and compute the MVN log density for each sample.

samples <- sparseMVN::rmvn.sparse(R, pm, CH, prec=TRUE)
logf <- sparseMVN::dmvn.sparse(samples, pm, CH, prec=TRUE)

The ability to accept a precision matrix, rather than having to invert it to a covariance matrix, is a valuable feature of sparseMVN. This is because the inverse of a sparse matrix is not necessarily sparse. In the following chunk, we invert the Hessian, and drop zero values to maintain any remaining sparseness. Note that there are 10,404 total elements in the Hessian.

Matrix::nnzero(H)
Hinv <- Matrix::drop0(solve(H))
Matrix::nnzero(Hinv)

Nevertheless, we should check that the log densities from dmvn.sparse() correspond to those that we would get from dmvnorm().

logf_dense <- mvtnorm::dmvnorm(samples, pm, as.matrix(Hinv), log=TRUE)
all.equal(logf, logf_dense)

Performance

In this section we show the efficiency gains from sparseMVN by comparing the run times between rmvn.sparse() and rmvnorm(), and between dmvn.sparse() and mvtnorm::dmvnorm. In these tests, we construct covariance/precision matrices with the same structure as the Hessian of the log posterior density of the example model in Section 2.1. Parameters are ordered such that the matrix has a block-arrow pattern (Figure @ref(fig:sparse-patterns), left).

The following table summarizes the case conditions. N and k refer, respectively, to the number of blocks in the block-arrow structure (analogous to heterogeneous units in the binary choice example), and the size of each block. The total number of variables is M=(N+1)k, and the total number of elements in the matrix is M2. The three rightmost columns present the number of nonzeros in the full matrix and lower triangle, and the sparsity (proportion of matrix elements that are nonzero. The size and sparsity of the test matrices vary through manipulation of the number of blocks (N), the size of each block (k), and the number of rows/columns in the margin (also k). Each test matrix has (N+1)k rows and columns.

Cases for the timing comparison.
nonzeros
k N variables elements full lower tri sparsity
2 10 22 484 124 73 0.256
2 20 42 1,764 244 143 0.138
2 50 102 10,404 604 353 0.058
2 100 202 40,804 1,204 703 0.030
2 200 402 161,604 2,404 1,403 0.015
2 300 602 362,404 3,604 2,103 0.010
2 400 802 643,204 4,804 2,803 0.007
2 500 1,002 1,004,004 6,004 3,503 0.006
4 10 44 1,936 496 270 0.256
4 20 84 7,056 976 530 0.138
4 50 204 41,616 2,416 1,310 0.058
4 100 404 163,216 4,816 2,610 0.030
4 200 804 646,416 9,616 5,210 0.015
4 300 1,204 1,449,616 14,416 7,810 0.010
4 400 1,604 2,572,816 19,216 10,410 0.007
4 500 2,004 4,016,016 24,016 13,010 0.006

Figure @ref(fig:densRand) compares mean run times to compute 1,000 MVN densities, and generate 1,000 MVN samples, using rmvn.sparse() and dmvn.sparse() from sparseMVN, and dmvnorm() and rmvnorm() in sparseMVN. Times were collected over 200 replications on a 2013-vintage Mac Pro with a 12-core 2.7 GHz Intel Xeon E5 processor and 64 GB of RAM. The times for mvtnorm are faster than sparseMVN for low dimensional cases (N50), but grow quadratically in the number of variables. This is because the number of elements stored in a dense covariance matrix grows quadratically with the number of variables. In this example, storage and computation requirements for the sparse matrix grow linearly with the number of variables, so the sparseMVN run times grow linearly as well (Braun and Damien 2016). The comparative advantage of sparseMVN increases with the sparsity of the covariance matrix.[^5]

Mean computation time for simulating 1,000 MVN samples, and computing 1,000 MVN densities, averaged over 200 replications. Densities were computed using [dmvnorm] and [dmvn.sparse()], while random samples were generated using [rmvnorm()] and [rmvn.sparse()].

Mean computation time for simulating 1,000 MVN samples, and computing 1,000 MVN densities, averaged over 200 replications. Densities were computed using [dmvnorm] and dmvn.sparse(), while random samples were generated using rmvnorm() and rmvn.sparse().

The sparseMVN functions always require a sparse Cholesky decomposition of the covariance or precision matrix, and the mvtnorm functions require a dense precision matrix to be inverted into a dense covariance matrix. Figure @ref(fig:cholSolve) compares the computation times of these preparatory steps. There are three cases to consider: inverting a dense matrix using solve(), decomposing a sparse matrix using Matrix::Cholesky(), and decomposing a dense matrix using chol(). Applying chol() to a dense matrix is not a required operation in advance of calling dmvnorm() or rmvnorm(), but those functions will invoke some kind of decomposition internally. We include it in our comparison because it comprises a substantial part of the computation time. The decomposition and inversion operations on the dense matrices grow cubically as the size of the matrix increases. The sparse Cholesky decomposition time is negligible. For example, the mean run time for the N=500, k=4 case is about 0.39 ms.

Computation time for Cholesky decomposition of sparse and dense matrices, and inversion of dense matrices.

Computation time for Cholesky decomposition of sparse and dense matrices, and inversion of dense matrices.

References

Bates, Douglas, and Martin Maechler. 2017. : Sparse and Dense Matrix Classes and Methods. https://CRAN.R-project.org/package=Matrix.
Braun, Michael, and Paul Damien. 2016. “Scalable Rejection Sampling for Bayesian Hierarchical Models.” Marketing Science 35 (3): 427–44. https://doi.org/10.1287/mksc.2014.0901.
Golub, Gene H, and Charles F Van Loan. 1996. Matrix Computations. 3rd ed. Johns Hopkins University Press.