Duncan's suggestion to time things is important -- and would make a very useful 
short communication or blog!
There are frequently differences of orders of magnitude in timing.

I'll also suggest that it is worth some crude timings of different solvers. 
There is sufficient variation
over problems that this won't decide definitively which solver is fastest, but 
you might eliminate one or
two that are poor for your situation. Depending on numbers of parameters, I'd 
guess ncg or its predecessor
Rcgmin will be relatively good. LBFGS variants can be good, but sometimes seem 
to toss up disasters. Most
of these can be accessed with optimx package to save coding. By removing some 
checks and safeguards in
optimx you could likely speed up things a bit too.

If full optimum is not needed, some attention to early stopping might be 
worthwhile, but I've seen lots
of silly mistakes made playing with tolerances, and if you go that route, 
choose a custom termination
rule that fits your particular problem or you'll get rubbish.

JN

On 2025-12-02 08:48, Duncan Murdoch wrote:
On 2025-12-02 7:41 a.m., Therneau, Terry M., Ph.D. via R-devel wrote:
I have a complex likelihood function f() to maximize, with lots of arguments (some of which set up indexes for derivatives, for instance). When using something like optim(), one can pass these arguments through via its � arg, or could make the likelihood function f() live in the same environment as the main routine so they are found directly.    Is there any advantage of one versus the other wrt speed?    At the end of the day, f() may get called thousands of times in a Hamiltonian MCMC.

Since R does not replicate arguments that are used in a read-only fashion, one might expect little to no penalty for having them on the call chain, unless the bookkeeping for copy-on-write is itself time consuming.

By the way, a nice way to put the args in the environment of the objective 
function is to use local() or a builder, e.g.

objective <- local({
   arg1 <- 1
   arg2 <- 2
   arg3 <- 3
   function(x) {
   # objective code here that can see arg1, arg2, arg3
   }
})

or

makeObjective <- function(arg1, arg2, arg3) {
   force(arg1) # evaluate the promises
   force(arg2)
   force(arg3)

   function(x) {
     # objective code here that can see arg1, arg2, arg3
   }
}

objective <- makeObjective(1,2,3)

Duncan Murdoch

______________________________________________
[email protected] mailing list
https://stat.ethz.ch/mailman/listinfo/r-devel

______________________________________________
[email protected] mailing list
https://stat.ethz.ch/mailman/listinfo/r-devel

Reply via email to