R/sparseMarkov.R

Defines functions runSparseMarkovChain

Documented in runSparseMarkovChain

#'
#'  sparseMarkov.R
#'
#'  discrete-time finite-state Markov chain simulation
#'  using a sparse matrix representation of the transition matrix
#'
#'  $Revision: 1.6 $ $Date: 2026/04/27 03:10:27 $
#'
#'  Copyright (c) Adrian Baddeley 2026
#'  GNU Public Licence (>= 2.0)


runSparseMarkovChain <- function(P, x0, nsteps, ..., 
                                 result=c("history", "last"),
                                 check=TRUE,
                                 method=c("C", "interpreted")) {
  method <- match.arg(method)
  result <- match.arg(result)
  savehistory <- (result == "history")
  x0 <- as.integer(x0)
  nx <- length(x0)
  if(method == "C") {
    ## prevent symmetric matrix from being represented by upper triangle
    P <- as(P, "generalMatrix")
    ## convert to row-major representation for use in Markov chain
    P <- as(P, "RsparseMatrix")
    if(!inherits(P, "RsparseMatrix"))
      stop("Unable to convert P to a sparse matrix in row-major form",
         call.=FALSE)
    if(isSymmetric(P) && .hasSlot(P, "uplo"))
      stop("Internal error: unable to suppress upper-triangular representation",
           call.=FALSE)
  }
  if(check) {
    ra <- range(P)
    if(!all(is.finite(ra)))
      stop("P contains infinite, NA or NaN entries", call.=FALSE)
    if(ra[1L] < 0) stop("P contains negative entries", call.=FALSE)
    if(ra[2L] > 1) stop("P contains entries greater than 1", call.=FALSE)
    rs <- range(rowSums(P))
    if(max(abs(rs-1)) > sqrt(.Machine$double.eps))
      stop("Some of the rows of P do not sum to 1", call.=FALSE)
    rx <- range(x0)
    if(rx[1L] < 1 || rx[2L] > nrow(P))
      stop("Some indices in x0 are out of range", call.=FALSE)
  }

  if(method == "C") {
    nentries <- length(P@x)
    if(nentries > .Machine$integer.max || is.na(as.integer(nentries)))
      stop("Sorry, long vectors are not yet supported", call.=FALSE)
  }
  
  ## >>>>>>>>>>>>  run chain  <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
  switch(method,
         interpreted = {
           x <- x0
           if(savehistory) history <- x0
           for(istep in 1:nsteps) {
             xnew <- integer(0)
             ## make one transition for each particle
             for(i in x) {
               p.i <- P[i, ]
               jj <- which(p.i > 0)
               p.ij <- p.i[jj]
               xnew <- c(xnew, sample(jj, 1L, prob=p.ij))
             }
             ## current state of each particle
             x <- xnew
             ## append to history (column = step, row = particle)
             if(savehistory) history <- cbind(history, x)
           }
         },
         C = {
           if(savehistory) {
             z <- .C(SP_rMCspMH,
                     nrows = as.integer(dim(P)[1L]),
                     nsparse = as.integer(length(P@x)),
                     probval = as.double(P@x),
                     colindex = as.integer(P@j),
                     rowstart = as.integer(P@p),
                     npoints = as.integer(nx),
                     startpos = as.integer(x0 - 1L),  # zero-based indices
                     nsteps = as.integer(nsteps),
                     history = as.integer(integer(nx * (nsteps + 1L))),
                     PACKAGE = "spatstat.sparse")
             history <- matrix(z$history + 1L, nx, nsteps+1L, byrow=TRUE)
           } else {
             z <- .C(SP_rMCspMF,
                     nrows = as.integer(dim(P)[1L]),
                     nsparse = as.integer(length(P@x)),
                     probval = as.double(P@x),
                     colindex = as.integer(P@j),
                     rowstart = as.integer(P@p),
                     npoints = as.integer(nx),
                     startpos = as.integer(x0 - 1L), # zero-based indices
                     nsteps = as.integer(nsteps),
                     endpos = as.integer(integer(nx)),
                     PACKAGE = "spatstat.sparse")
             x <- z$endpos + 1L
           }
         })
  if(savehistory) {
    dimnames(history) <- NULL
    if(nx == 1) history <- history[1L, ]
    return(history)
  }
  return(x)
}


  

Try the spatstat.sparse package in your browser

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

spatstat.sparse documentation built on May 21, 2026, 9:07 a.m.