python scipy.optimize 求解带边界约束的非线性规划报错“Positive directional derivative for linesearch”

问题背景

在数据处理过程中,需要求解形如下图的非线性规划:
  在这里插入图片描述
  (在我的学习任务中不等式约束边界会变化,上图示意一个特殊情况)

求解方法一:scipy.optimize

尝试利用scipy.optimize中的.minimize()函数进行求解,将等式与不等式约束存入cons字典中。

from scipy.optimize import minimize
import numpy as np

fun = lambda x : x[0] + x[1] + x[2]# 约束函数
cons = ({'type': 'eq', 'fun': lambda x: x[0]**2+ x[1]**2+ x[2]**2 - 1}, # xyz=1

{'type': 'ineq', 'fun': lambda x: 1-x[0]**2 },#  1 >= x**2
{'type': 'ineq', 'fun': lambda x: x[0]**2-1 },#  x**2 >=1

{'type': 'ineq', 'fun': lambda x: 0-x[1]**2 },
{'type': 'ineq', 'fun': lambda x: x[1]**2- 0 },

{'type': 'ineq', 'fun': lambda x:  0-x[2]**2 },
{'type': 'ineq', 'fun': lambda x: x[2]**2-0 },

{'type': 'ineq', 'fun': lambda x: 1-x[0]**2-x[1]**2 },
{'type': 'ineq', 'fun': lambda x: x[0]**2+x[1]**2 -1},

{'type': 'ineq', 'fun': lambda x: 1-x[0]**2-x[2]**2 },
{'type': 'ineq', 'fun': lambda x: x[0]**2+x[2]**2-1}{'type': 'ineq', 'fun': lambda x: 0-x[1]**2-x[2]**2 },
{'type': 'ineq', 'fun': lambda x: x[1]**2+x[2]**2- 0 }
)
b1=(0,1)
b2=(0,1)
b3=(0,1)
bonds=(b1,b2,b3)
x0 = np.array((0.5, 0.5, 0.5)) # 设置初始值
res = minimize(fun, x0, method='SLSQP', constraints=cons,bounds=bonds, 
              tol=None)
#print(np.around(res.x,4))#量子形式的系数
print('最小值:',res.fun)
print('最优解:',res.x)
print('迭代终止是否成功:', res.success)
print('迭代终止原因:', res.message)

运行结果:
在这里插入图片描述
问题:

  1. 明显该规划问题的最优解为(1, 0, 0),但运行结果计算错误。通过查看具体运行信息,可以发现迭代终止并未成功,并给出了‘Positive directional derivative for linesearch’信息;
  2. 该规划需要对初始值进行声明,即x0 = np.array((0.5, 0.5, 0.5))部分,且初始值的设定对运行结果影响很大,若设x0 = np.array((1, 0, 0)),则运行可以成功,明显不符合一般计算逻辑。

其他:
  尝试过一些其他解决方法,指路Positive directional derivative for linesearch的四种解决方案Python 的 scipy.optimize.minimize 与 SLSQP 失败并显示 “Positive directional derivative for linesearch”,均无法成功运行。最终尝试替换计算方法,使用差分进化算法进行求解(指路差分进化算法解决有约束优化问题(Python)),具体过程如下:

求解方法二:差分进化算法

from sko.DE import DE

def obj_func(p):
    x1, x2, x3 = p
    return x1 + x2  + x3 

constraint_eq = [lambda x: 1 -x[0]**2 - x[1]**2 - x[2]**2]

constraint_ueq = [lambda x: 1-x[0]**2, lambda x: x[0]**2-1,
                  lambda x: 0-x[1]**2,lambda x: x[1]**2- 0,
                  lambda x: 0-x[2]**2,lambda x: x[2]**2- 0,
                  lambda x: 1-x[0]**2-x[1]**2,lambda x: x[0]**2+x[1]**2 -1,
                  lambda x: 1-x[0]**2-x[2]**2,lambda x: x[0]**2+x[2]**2-1,
                  lambda x: 0-x[1]**2-x[2]**2,lambda x: x[1]**2+x[2]**2- 0                
                 ]
de = DE(func=obj_func, n_dim=3, size_pop=50, max_iter=100, lb=[0, 0, 0], ub=[1, 1, 1],
        constraint_eq=constraint_eq, constraint_ueq=constraint_ueq)
best_x, best_y = de.run()

print('best_x:', np.around(best_x,2), '\n', 'best_y:', best_y)

运行结果:
运行结果

总结

从计算结果来看,差分进化算法不但不需要指定初始值,也可以成功计算一些特殊情况,此外针对更一般的边界约束情况,上述规划仍可成功计算。因此方法二更适用于本问题。

  • 9
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Python scipy.optimizePython科学计算库scipy中的一个模块,它提供了许多用于最小化或寻找方程根的优化算法。这些算法包括无约束约束的优化、全局优化、最小二乘拟合以及曲线拟合等。它们可用于处理各种问题,如最小化成本、最大化效益、最大化收益等。scipy.optimize模块还支持并行计算和大规模优化问题的求解。无论是初学者还是专业人士,都可以使用Python scipy.optimize来解决他们的优化问题。 ### 回答2: Python中的Scipy.optimize是一个优化算法库,用于解决各种数学优化问题。这个库包含了一系列优化算法,可以帮助用户优化一些目标函数。 Scipy.optimize包含了许多优化算法,其中的一些算法包括最小二乘法、有约束优化、全局优化、非线性方程求解、拟合等。由于优化问题的多样性,使用不同的算法可能会得到不同的结果,因此在选择算法时需要理解问题本身,并进行选择。 优化算法的核心是寻找一个局部最优解或全局最优解。这一过程是通过迭代方法来完成的,迭代过程中的每个步骤都可以用优化算法中的不同方法来处理。不同的算法对优化目标函数的形式和要求有不同的处理方式。使用Scipy.optimize库可以方便地调用这些算法,这些算法经过优化和测试,是最常用的算法。 除了包含不同类型的优化算法,Scipy.optimize还集成了许多其他的辅助工具,例如最优解计算、目标函数的梯度和海森矩阵计算等。 总体而言,Scipy.optimize是一个功能强大的优化算法库,它可以处理各种数学优化问题,包括有约束和无约束问题,凸和非凸问题等等。此外,Scipy.optimize库还提供了许多实用的函数,例如求解目标函数的一阶和二阶导数的函数以及许多常用优化算法,使优化过程更高效、更方便。因此,Scipy.optimizepython优化问题求解的重要库之一。 ### 回答3: Python中的Scipy.optimize模块是一个专门用于求解优化问题的工具包。该模块提供了多种优化算法来最小化或最大化目标函数,并且能够针对不同类型的问题进行使用。 Scipy.optimize模块中包含了多个优化算法,其中大多数算法都用于非线性优化问题。这些算法包括:Nelder-Mead、Powell、CG、BFGS、L-BFGS-B和TNC等。每个算法都有自己的优缺点,并且特别适合不同类型的优化问题。 Scipy.optimize模块的另一个功能是提供了一些特定问题的专用算法。其中包括线性规划、非线性规划、二次规划、整数规划、全局优化等算法。这里有一个例子:线性规划可以用于解决最小化或最大化线性方程组的问题。 在使用Scipy.optimize模块时,需要指定目标函数和隐式约束条件,并且还需要传递初始点以作为算法开始优化的初始值。例如: from scipy.optimize import minimize import numpy as np #定义目标函数 def f(x): return np.sum(x**2) #定义约束条件 #x.sum() - 1 = 0, x_i >= 0, for i=1,...,n cons = ({'type': 'eq', 'fun': lambda x: np.sum(x) - 1.0}, {'type': 'ineq', 'fun': lambda x: x }) #初始点 x0 = [0.25, 0.25, 0.25, 0.25] #最小化方法 res = minimize(f, x0, method='SLSQP', constraints=cons) 在这个例子中,我们定义了一个目标函数f(x) = x1^2 + x2^2 + x3^2 + x4^2,并且只有一个约束条件x1 + x2 + x3 + x4 = 1和所有变量都必须大于等于0。使用Scipy.optimize模块的minimize函数进行优化,以最小化目标函数,这里使用了SLSQP算法。函数的返回值是最优点及其函数值。 总之,PythonScipy.optimize模块是一个广泛使用的工具包,可以用于求解多种类型的优化问题。它与其他Python的科学计算工具(如Numpy和Matplotlib等)很好地集成在一起,可以在整个Python科学计算生态系统中广泛使用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值