我尝试使用SciPy中实现的SHGO算法,但是当目标函数使用多个参数时,我遇到了问题。如果我正确理解了错误,我没有将额外的参数传递给目标函数,但是我看不出语法哪里有错。有人能解释一下错误的根本原因是什么以及如何修复吗?
下面是我所面临的问题的一个可重复的例子。
import numpy as np
import scipy.optimize as opt
def fobj(x, y, z):
return (x+y+z).sum()
x0 = np.array([0.5, 0.5, 0.5, 0.5])
y = np.array([1, 3, 5, 7])
z = np.array([10, 20, 30, 40])
bnds = list(zip([0, 1, 2, 3], [2, 3, 4, 5]))
cons = {'type': 'eq', 'fun': lambda x: x.sum() - 14}
min_kwargs = {'method': 'SLSQP', 'options': {'maxiter': 100, 'disp': True}}
ret = opt.shgo(func=fobj, bounds=bnds, args=(y, z), constraints=cons, minimizer_kwargs=min_kwargs, options={'disp': True})
运行时,将显示以下追溯。
Splitting first generation
Traceback (most recent call last):
File "C:\path\lib\site-packages\scipy\optimize\_shgo_lib\triangulation.py", line 630, in __getitem__
return self.cache[x]
KeyError: (0, 0, 0, 0)
During handling of the above exception, another exception occurred:
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "C:\path\lib\site-packages\scipy\optimize\_shgo.py", line 420, in shgo
shc.construct_complex()
File "C:\path\lib\site-packages\scipy\optimize\_shgo.py", line 733, in construct_complex
self.iterate()
File "C:\path\lib\site-packages\scipy\optimize\_shgo.py", line 876, in iterate
self.iterate_complex()
File "C:\path\lib\site-packages\scipy\optimize\_shgo.py", line 895, in iterate_hypercube
self.HC = Complex(self.dim, self.func, self.args,
File "C:\path\lib\site-packages\scipy\optimize\_shgo_lib\triangulation.py", line 25, in __init__
self.n_cube(dim, symmetry=symmetry)
File "C:\path\lib\site-packages\scipy\optimize\_shgo_lib\triangulation.py", line 76, in n_cube
self.C0.add_vertex(self.V[origintuple])
File "C:\path\lib\site-packages\scipy\optimize\_shgo_lib\triangulation.py", line 634, in __getitem__
xval = Vertex(x, bounds=self.bounds,
File "C:\path\lib\site-packages\scipy\optimize\_shgo_lib\triangulation.py", line 557, in __init__
self.f = func(x_a, *func_args)
File "C:\path\lib\site-packages\scipy\optimize\_optimize.py", line 466, in function_wrapper
fx = function(np.copy(x), *(wrapper_args + args))
TypeError: fobj() takes 3 positional arguments but 5 were given
我不明白为什么会引发TypeError。它是说传递给目标函数fobj
的参数是5个而不是3个,但据我所知,我只传递了(y, z)
,所以我看不出它们怎么会是5个!
请注意,我也尝试过将局部极小化字典重写为min_kwargs = {'method': 'SLSQP', 'args': (x0), 'options': {'maxiter': 100, 'disp': True}}
,但我一直面临着同样的错误。我确信我传递的参数不正确,但我不知道如何正确地做。任何帮助将不胜感激。
我使用的是:Python 3.10.5、Numpy 1.22.4和SciPy 1.8.1。
2条答案
按热度按时间ff29svar1#
正如您所发现的,问题是bug in
shgo
。你可以通过避免使用
args
参数来避免shgo
中的bug。不要使用args
,而是使用一个fobj
的 Package 器来捕获闭包中的y
和z
。一个简单的方法是使用lambda
表达式:将shgo
的第一个参数从func=fobj
更改为func=lambda x, y=y, z=z: fobj(x, y, z)
,并从调用中删除args
参数。kr98yfug2#
问题是
sum
是一个方法,而不是数组的属性。如果你把目标函数
此外,在开发版本上有一个bug(很快就会更新)。
该方法似乎也不是很健壮,对于sum(x)= 14,它只有一个解,这可能是一个问题,因为有效的搜索空间是0维的,但即使我将其放宽到sum(x)= 12这样的情况,该方法也不能给予解。
下面是一个更完整的工作示例