R/utils.R

Defines functions whichAllInBag nobs.default hasMethod hasComponent doCall defaultFitNames defaultNames dataSubset combineResults combineData checkSelect addNames.default addNames.list addNames

# --------------------------------------
# Author: Andreas Alfons
#         Erasmus Universiteit Rotterdam
# --------------------------------------

# add default names for prediction error results
addNames <- function(x) UseMethod("addNames")
#addNames.list <- function(x) lapply(x, addNames)         # throws error
addNames.list <- function(x) lapply(x, addNames.default)  # workaround
addNames.default <- function(x) {
    if(is.null(p <- ncol(x))) {
        if(is.null(names(x))) names(x) <- defaultNames(length(x))
    } else {
        if(is.null(colnames(x))) colnames(x) <- defaultNames(p)
    }
    x
}

# check selection indices for subsets
checkSelect <- function(select = NULL, names, returnNames = TRUE) {
    all <- seq_along(names)
    names(all) <- names               # works for characters
    select <- unique(all[select])     # remove duplicates
    select <- select[!is.na(select)]  # remove nomatches
    if(returnNames) names[select] else select
}

# combine data (used for predictions from PE folds)
combineData <- function(x, drop = TRUE) {
    if(drop && is.null(dim(x[[1]]))) {
        unlist(x)
    } else do.call(rbind, x)
}

# combine prediction error results from a list of models
combineResults <- function(x, fits = names(x)) {
    # initializations
    m <- length(x)
    if(is.null(fits)) fits <- defaultFitNames(m)
    else if(any(i <- fits == "")) fits[i] <- defaultFitNames(m)[i]
    if(!is.numeric(fits)) fits <- factor(fits, levels=fits)
    # combine prediction errors and standard errors
    pe <- combineData(lapply(x, "[[", "pe"), drop=FALSE)
    pe <- data.frame(Fit=fits, pe, row.names=NULL)
    se <- combineData(lapply(x, "[[", "se"), drop=FALSE)
    se <- data.frame(Fit=fits, se, row.names=NULL)
    out <- list(pe=pe, se=se)
    # combine results from all replications if available
    reps <- combineData(lapply(x, "[[", "reps"))
    if(!is.null(reps)) {
        R <- nrow(reps) / length(fits)
        out$reps <- data.frame(Fit=rep(fits, each=R), reps, row.names=NULL)
    }
    # return list of combined results
    out
}

# retrieve data subsets
dataSubset <- function(x, i, drop = FALSE) {
    if(is.null(dim(x))) {
        x[i]
    } else x[i, , drop=FALSE]
}

# replace data subsets
"dataSubset<-" <- function(x, i, value) {
    if(is.null(dim(x))) {
        x[i] <- value
    } else x[i, ] <- value
    x
}

# default names for prediction error results
defaultNames <- function(p) {
    if(p == 1) {
        "PE"
    } else if(p > 0) {
        paste("PE", seq_len(p), sep="")
    } else character()
}

# default names for model fits
defaultFitNames <- function(m) {
    if(m == 1) {
        "Fit"
    } else if(m > 0) {
        paste("Fit", seq_len(m), sep="")
    } else character()
}

## call a function by either
# 1) simply evaluating a supplied function for the basic arguments if there are
#    no additional arguments in list format
# 2) evaluating a supplied function with 'do.call' if there are additional
#    arguments in list format
doCall <- function(fun, ..., args = list()) {
    if(length(args) == 0) {
        fun(...)
    } else do.call(fun, c(list(...), args))
}

# check if a list or object has a certain component
hasComponent <- function(x, name) name %in% names(x)

# check if a generic function has a method for a certain class
# function name needs to be supplied instead of the function itself
#' @importFrom utils getS3method
hasMethod <- function(fun, class) {
    !is.null(getS3method(fun, class, optional=TRUE))
}

# retrieve the number of observations
nobs.default <- function(object, ...) {
    n <- nrow(object)                   # matrix or data.frame
    if(is.null(n)) n <- length(object)  # vector
    n
}

# find which bootstrap samples have all observations in the bag
whichAllInBag <- function(n, samples) {
    indices <- seq_len(n)
    m <- apply(samples, 2, function(i) length(indices[-i]))  # number out-of-bag
    which(m == 0)
}

Try the perry package in your browser

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

perry documentation built on Nov. 3, 2021, 5:08 p.m.