The Statistician's Apprentice: An Introduction to the SWP Operator
The sweep operator as defined in (Dempster, 1969), commonly referred to as the SWP operator, is a useful tool for a computational statistician working with covariance matrices. In particular, the SWP operator allows a statistician to quickly regress all variables against one specified variable, obtaining OLS estimates for regression coefficients and variances in a single application. Subsequent applications of the SWP operator allows for regressing against more variables.
In this blog post, I will define the sweep operator, provide an application of the sweep data to simulated data, and provide some references for further study. Examples will be provided in R through the ISR3
package (Lisic, 2016).
The basic SWP operator is parameterized by a matrix and a set of indices associated with rows and columns of the matrix being swept. E.g. to sweep the matrix by the index, such that , the operator has the form . The operator may also be written more descriptively by listing the elements of as . The application of the SWP operator to the row and column of produces a new matrix with elements
As an example, consider the application of the SWP operator to the first row and column of the covariance matrix for the random variables ,
How is this useful? Well, if we substitute the covariance matrix with the sample covariance matrix, where is an estimator of and
Example
Lets try this with some real data. I have used the SWP function included with the ISR3
package.
library(ISR3) set.seed(12) n <- 100 p <- 3 # generate a positive definite matrix for covariance Sigma <- rWishart(1,p+1,diag(p))[,,1] Sigma_inv <- chol2inv(chol(Sigma)) Sigma_inv_chol <- chol(Sigma_inv) # generate 'n' multivate normal deviates X <- Sigma_inv_chol %*% matrix(rnorm(n*p),nrow=p) X <- t(X) colnames(X) <- sprintf("X_%d",1:p) XX <- t(X) %*% X #Sweep by the first row/column of XX SWP_1 <- SWP(XX,1) SWP_1
## X_1 X_2 X_3 ## X_1 -0.002381092 0.1375626 -0.01674788 ## X_2 0.137562633 50.0987932 -4.95181624 ## X_3 -0.016747875 -4.9518162 41.73979419
# linear models fit_21 <- lm(X_2 ~ -1 + X_1, data=as.data.frame(X)) fit_31 <- lm(X_3 ~ -1 + X_1, data=as.data.frame(X)) # SLR coefficents for X_1 print(fit_21$coefficients)
## X_1 ## 0.1375626
print(fit_31$coefficients)
## X_1 ## -0.01674788
# sum of squares print(sum(fit_21$residuals^2))
## [1] 50.09879
print(sum(fit_31$residuals^2))
## [1] 41.73979
# covariance between X_2 and X_3 given X_1 print(sum(fit_21$residuals * fit_31$residuals))
## [1] -4.951816
# negative inverse of the covariance matrix of X_1 print( -1/((n-1)*var(X[,1])) )
## [1] -0.002381293
lm
match up to our swept results. Likewise, the diagonal terms for the second and third column are equivalent to the sum of squares or the residuals. The remaining unmatched term is just the negative inverse of the covariance matrix of unscaled by .
A subsequent sweep on the second rows and columns produces the matrix,
lm
to our SWP operator.
A final sweep by the third row and column produces , the negative inverse of the unscaled covariance matrix of . This can be quickly verified through R
:
SWP(XX,1:3) + chol2inv(chol(XX))
## X_1 X_2 X_3 ## X_1 0.000000e+00 0.000000e+00 1.355253e-20 ## X_2 0.000000e+00 3.469447e-18 8.673617e-19 ## X_3 1.355253e-20 8.673617e-19 0.000000e+00
Undo!
We can also undo sweeps with the reverse sweep (RSWP) operator through the following element-wise operations
What Next?
A few good references on the subject beyond (Dempster, 1969) can be found in (Goodknight, 1979) and (Little and Rubin, 2014). There are a few slight differences between these implementations, but all generally do the same thing.
One function that doesn't do the same thing is the sweep
function in R
. This function can be used to write a SWP implementation, but doesn't sweep by itself.
Code examples can be found in the ISR3
package on CRAN
and in lots of other packages.
Examples in C
can be found in ISR3
and as an internal function in the MNP
package (Imai and Van Dyk, 2013).
Enjoy your sweeping, next time I'll go over sweeping with the generalized inverse!
References
Dempster, A.P. (1969). Elements of continuous multivariate analysis. Reading, MA: Addison-Wesley.
Goodnight, J. H. (1979). A tutorial on the SWEEP operator. The American Statistician, 33(3), 149-158.
Imai, K., and Van Dyk, D. A. (2013). MNP: R Package for fitting the Multinomial Probit Model, R package version 2.6-4, https://CRAN.R-project.org/package=MNP
Lisic, J. J. (2016). ISR3: Iterative Sequential Regression, R package version 0.98, https://CRAN.R-project.org/package=ISR3
Little, R. J., and Rubin, D. B. (2014). Statistical analysis with missing data. John Wiley & Sons.