inst/examples/brockett/rproblem/driver-withhessian.R

library(ManifoldOptim)
source("../../test_util.R")

set.seed(1234)

p <- 5
n <- 15
B <- matrix(rnorm(n*n), nrow=n)
B <- B + t(B) # force symmetric
D <- diag(p:1, p)

# Variables within the optimizer are treated as a single vector.
# tx is a helper function that transforms the optimization variable
# into something we can directly use.
tx <- function(x) {
	matrix(x, n, p)
}
f <- function(x) {
	X <- tx(x)
	Trace( t(X) %*% B %*% X %*% D )
}
g <- function(x) {
	X <- tx(x)
	2 * B %*% X %*% D
}

## Computes 2 * (D %x% B) %*% eta
## without explictly computing the large & sparse D %x% B matrix
h <- function(x, eta) {
	z <- B %*% matrix(eta, n, p)
	for (i in 1:p) {
		z[,i] <- 2 * D[i,i] * z[,i]
	}
	as.numeric(z)
}

h <- function(x, eta) { 2 * (D %x% B) %*% eta }

# 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, g, h)

X0 <- orthonorm(matrix(rnorm(n*p), nrow=n, ncol=p))
x0 <- as.numeric(X0)
prob$objFun(x0)			# Test the obj fn
head(prob$gradFun(x0))	# Test the grad fn

mani.params <- get.manifold.params(IsCheckParams = FALSE)
solver.params <- get.solver.params(Max_Iteration = 1000,
	IsCheckGradHess = TRUE) 
mani.defn <- get.stiefel.defn(n, p)

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

# Compare to closed-form solution
eig <- eigen(B)
X.star <- eig$vectors[,seq(n,n-p+1)]
f(res$xopt)
f(as.numeric(X.star))

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.