如何调试scipy中的`OverflowError: math range error`

0 投票
1 回答
3540 浏览
提问于 2025-04-18 12:56

我有一段测试代码,目的是计算一个最大似然估计(MLE)。

import numpy as np
from scipy.optimize import minimize


def loglikelihood(params, data):
    (mu, alpha, beta) = params
    tlist = np.array(data)
    r = np.zeros(len(tlist)+1)
    for i in xrange(2,len(tlist)):
        r[i] = math.exp(-beta*(tlist[i]-tlist[i-1]))*(1+r[i-1])
    loglik  = data[-1]*mu
    loglik = loglik+alpha/beta*sum(np.exp(-beta*(tlist[-1]-tlist))-1)
    loglik = loglik+np.sum(np.log(mu+alpha*r))
    return -loglik


atimes=[58.98353497,   59.28420225,   59.71571013,   60.06750179,   61.24794134,
   61.70692463,   61.73611983,   62.28593814,   62.51691723,   63.17370423
,   63.20125152,   65.34092403,  214.24934446,  217.0390236,   312.18830525,
  319.38385604,  320.31758188,  323.50201334,  323.76801537,  323.9417007]

print minimize(loglikelihood, (0.01, 0.5,0.6), (atimes,))

在执行 res = minimize(loglikelihood, (0.01, 0.5,0.6), args = (atimes,)) 时,我遇到了 OverflowError: math range error 的错误。

我该如何解决这个问题呢?


我只是想把下面的最大似然估计(MLE)R语言代码翻译成Python,并用我放在atimes里的数据进行测试。

neg.loglik <- function(params, data, opt=TRUE) {
  mu <- params[1]
  alpha <- params[2]
  beta <- params[3]
  t <- sort(data)
  r <- rep(0,length(t))
  for(i in 2:length(t)) {
    r[i] <- exp(-beta*(t[i]-t[i-1]))*(1+r[i-1])
  }
  loglik <- -tail(t,1)*mu
  loglik <- loglik+alpha/beta*sum(exp(-beta*(tail(t,1)-t))-1)
  loglik <- loglik+sum(log(mu+alpha*r))
  if(!opt) {
    return(list(negloglik=-loglik, mu=mu, alpha=alpha, beta=beta, t=t,
                r=r))
  }
  else {
    return(-loglik)
  }
}

# insert your values for (mu, alpha, beta) in par
# insert your times for data
opt <- optim(par=c(1,2,3), fn=neg.loglik, data=data)

我尝试把Python代码改成返回loglik而不是-loglik。虽然不太清楚为什么R代码是要最大化,但我在http://docs.scipy.org/doc/scipy/reference/tutorial/optimize.html上没有找到明确的解释。

现在我得到了

RuntimeWarning: invalid value encountered in log
loglik = loglik+np.sum(np.log(mu+alpha*r))

1 个回答

1

你对R语言函数的翻译有错误,主要有两个问题:(i) 一是索引从1开始和从0开始的区别,(ii) 二是一些符号错误。

因此,这个函数可能没有下限,也就是说没有最小值。优化器会试着在函数值下降的方向上,使用越来越大的参数值,结果就把很大的数字放进了math.exp,导致溢出。

这个溢出错误不是来自scipy,而是来自你自己写的函数。

撰写回答