R/exp_parallel.R

Defines functions mean.exp_parallel sampler.exp_parallel surv.exp_parallel exp_parallel

Documented in exp_parallel mean.exp_parallel sampler.exp_parallel surv.exp_parallel

# ==========================================================================
# Closed-form parallel of exponential components
# ==========================================================================
#
# A parallel system of m independent exponentials with rates lambda_j has
#   F_sys(t) = prod_j (1 - exp(-lambda_j t))
#   S_sys(t) = 1 - F_sys(t)
# For iid components, E[max] = (1/lambda)(1 + 1/2 + ... + 1/m).
# For the heterogeneous case, E[max] = sum over nonempty subsets S of
# (-1)^(|S|+1) / sum(rates[S]) via inclusion-exclusion.
# ==========================================================================


#' Parallel of exponential components (closed form)
#'
#' Constructs a `dist_structure` representing a parallel system whose
#' components are independent exponentials. Closed-form methods are
#' provided for `surv`, `cdf`, `sampler`, and `mean` (the last via
#' inclusion-exclusion over the `2^m - 1` non-empty component subsets).
#'
#' @param rates Positive numeric vector of length `m`.
#' @return
#' `exp_parallel()` returns an object of class
#'   `c("exp_parallel", "parallel_dist", "coherent_dist", "dist_structure",
#'   "univariate_dist", "continuous_dist", "dist")`.
#'
#' The associated S3 methods return:
#' - `surv()`: a closure `function(t, ...)`.
#' - `cdf()` is derived via the `dist_structure` default and returns
#'   a closure `function(t, ...)` equal to `1 - surv(x)(t)`.
#' - `sampler()`: a closure `function(n, ...)` returning `n` random
#'   variates from the system lifetime distribution.
#' - `mean()`: a numeric scalar (the mean system lifetime,
#'   computed in closed form via inclusion-exclusion).
#' @examples
#' sys <- exp_parallel(c(1, 2, 3))
#' algebraic.dist::surv(sys)(1)
#' @export
exp_parallel <- function(rates) {
  stopifnot(is.numeric(rates), length(rates) >= 1L, all(rates > 0))
  components <- lapply(rates, algebraic.dist::exponential)
  obj <- parallel_dist(components)
  obj$rates <- as.numeric(rates)
  class(obj) <- c("exp_parallel", class(obj))
  obj
}


#' @rdname exp_parallel
#' @param x An `exp_parallel` object.
#' @param ... Ignored.
#' @export
surv.exp_parallel <- function(x, ...) {
  rates <- x$rates
  function(t, ...) {
    vapply(t, function(ti) 1 - prod(1 - exp(-rates * ti)), numeric(1L))
  }
}


#' @rdname exp_parallel
#' @export
sampler.exp_parallel <- function(x, ...) {
  samplers <- make_component_samplers(stats::rexp, rate = x$rates)
  function(n, ...) {
    apply(sample_component_matrix(samplers, n), 1L, max)
  }
}


#' @rdname exp_parallel
#' @export
mean.exp_parallel <- function(x, ...) {
  rates <- x$rates
  m <- length(rates)
  # Fast path: iid components. E[max of m iid Exp(lambda)] = H_m / lambda.
  if (length(unique(rates)) == 1L) {
    return(sum(1 / seq_len(m)) / rates[[1L]])
  }
  # General heterogeneous case: inclusion-exclusion over 2^m - 1 subsets.
  total <- 0
  for (sz in seq_len(m)) {
    subsets <- utils::combn(m, sz, simplify = FALSE)
    for (S in subsets) {
      total <- total + (-1)^(sz + 1L) / sum(rates[S])
    }
  }
  total
}

Try the dist.structure package in your browser

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

dist.structure documentation built on May 13, 2026, 1:07 a.m.