R/hccm.R

Defines functions hccm.default hccm

Documented in hccm hccm.default

#-------------------------------------------------------------------------------
# Revision history:
# 2009-09-16: optionally allow models with aliased coefficients J. Fox
# 2012-04-04: modified to allow weighted linear models. J. Fox
# 2020-06-25: Fix bug in hccm.lm() when model matrix includes just one column 
#             (reported by Justin Yap). J. Fox
# 2021-07-29: Report error when any hatvalue = 1 for all but hc0 and hc1
#             (following report of problem reported by Peng Ding) J. Fox
# 2022-10-11: Modified error reports for hatvalue = 1, and add for
#             hc0 and hc1. S. Weisberg
# 2022-11-18  fixed error messages
#-------------------------------------------------------------------------------

# Heteroscedasticity-corrected standard errors (Huber/White adjustment) (J. Fox)

hccm <- function(model, ...){
	UseMethod("hccm")
}

hccm.lm <-function (model, type = c("hc3", "hc0", "hc1", "hc2", "hc4"), 
		singular.ok = TRUE, ...) {
	e <- na.omit(residuals(model))
	removed <- attr(e, "na.action")
	wts <- if (is.null(weights(model))) 1 
			else weights(model)
	type <- match.arg(type)
	if (any(aliased <- is.na(coef(model))) && !singular.ok) 
		stop("there are aliased coefficients in the model")
	sumry <- summary(model, corr = FALSE)
	s2 <- sumry$sigma^2
	V <- sumry$cov.unscaled
	if (type == FALSE) 
		return(s2 * V)
	h <- hatvalues(model)
	if (!is.null(removed)){
		wts <- wts[-removed]
		h <- h[-removed]
	}
# error checking 1:  check if any elements of h are > 1- sqrt(machine.eps)
	bad <- which(h > 1 - sqrt(.Machine$double.eps))
	if(length(bad) > 0){
	  nms <- names(residuals(model))
	  stop("hatvalues (leverages) equal to or very close to 1 for the following cases:\n", 
	       if(length(bad) <=  10) paste(nms[bad], collapse=", ") else
	         paste0(paste(nms[bad[1:10]], collapse=", "), ", ..."),"\n",
	       "For type = 'hc1', 'hc2', or 'hc3' the hccm is undefined.\n",
	       "For type = 'hc0' or 'hc4' the hccm is singular and therefore\n",
	       "not a consistent estimator of the coefficient covariance matrix.")
	}
# end error checking 1
	X <- model.matrix(model)[, !aliased, drop=FALSE]
	df.res <- df.residual(model)
	n <- length(e)
	e <- wts*e
	p <- ncol(X)
	factor <- switch(type, hc0 = 1, hc1 = df.res/n,
	             hc2 = 1 - h, hc3 = (1 - h)^2, hc4 = (1 - h)^pmin(4, n * h/p))
	V <- V %*% t(X) %*% apply(X, 2, "*", (e^2)/factor) %*% V
# error checking 2:  Is V singular?
	if (qr(V)$rank < p)
	  stop("hccm estimator is singular with rank ", qr(V)$rank, 
	       ".\nIt is inconsistent as an estimate of the covariance matrix\nof the regression coefficients, which has rank ", p, 
	       ".\nThis is caused by ill-conditioning of the model matrix.")
	V
}

hccm.default<-function(model, ...){
	stop("requires an lm object")
}

Try the car package in your browser

Any scripts or data that you put into this service are public.

car documentation built on March 31, 2023, 6:51 p.m.