inst/examples/normal_mle_sphere/rproblem/driver.R

library(ManifoldOptim)
library(mvtnorm)

set.seed(1234)

# Try to estimate jointly: Sigma in SPD manifold and mu in sphere manifold
n <- 400
p <- 3
mu.true <- rep(1/sqrt(p), p)
Sigma.true <- diag(2,p) + 0.1
y <- rmvnorm(n, mean = mu.true, sigma = Sigma.true)

tx <- function(x) {
	idx.mu <- 1:p
	idx.S <- 1:p^2 + p
	mu <- x[idx.mu]
	S <- matrix(x[idx.S], p, p)
	# S <- (S + t(S)) / 2
	list(mu = mu, Sigma = S)
}
if (FALSE) {
  f <- function(x) {
  	par <- tx(x)
  	-sum(dmvnorm(y, mean = par$mu, sigma = par$Sigma, log = TRUE))
  }
} else {
  f <- function(x) {
    par <- tx(x)
    logdet <- determinant(par$Sigma, logarithm = TRUE)
    ll <- -0.5 * n * p * log(2*pi) - 0.5 * n * as.numeric(logdet$sign * logdet$modulus)
    for (i in 1:n) {
      q_i = y[i,] - par$mu
      ll <- ll - 0.5 * t(q_i) %*% solve(par$Sigma, q_i)
    }
    return(-ll)
  }
}

# Take the problem above (defined in R) and make a Problem object for it.
# The Problem can be passed to the optimizer in C++
mod <- Module("ManifoldOptim_module", PACKAGE = "ManifoldOptim")
prob <- new(mod$RProblem, f)

mu0 <- diag(1, p)[,1]
Sigma0 <- diag(1, p)
x0 <- c(mu0, as.numeric(Sigma0))

# Test the obj and grad fn
prob$objFun(x0)
# head(prob$gradFun(x0))

mani.defn <- get.product.defn(get.sphere.defn(p), get.spd.defn(p))
mani.params <- get.manifold.params(IsCheckParams = TRUE)
solver.params <- get.solver.params(DEBUG = 0, Tolerance = 1e-4,
	Max_Iteration = 1000, IsCheckParams = TRUE, IsCheckGradHess = FALSE)
deriv.params <- get.deriv.params(EpsNumericalGrad = 1e-8, EpsNumericalHessEta = 1e-4)

res <- manifold.optim(prob, mani.defn, method = "LRBFGS", 
	mani.params = mani.params, solver.params = solver.params,
	deriv.params = deriv.params, x0 = x0)
print(res)
print(tx(res$xopt))

Try the ManifoldOptim package in your browser

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

ManifoldOptim documentation built on Dec. 15, 2021, 1:07 a.m.