新闻中心
Scipy优化中多重线性约束的正确实现与性能优化

本文深入探讨了在`scipy.optimize.minimize`中使用多重线性约束时可能遇到的问题及其解决方案。文章首先揭示了Python中lambda函数与循环结合时常见的“延迟绑定”陷阱,并提供了两种修复方法。更重要的是,教程强调并演示了如何利用`scipy.optimize.LinearConstraint`这一专业工具,以显著提升线性约束优化问题的性能和准确性,为数值优化提供了最佳实践。
在数值优化问题中,特别是在使用scipy.optimize.minimize进行非线性规划(NLP)时,经常需要处理各种约束条件。线性约束因其结构简单和计算效率高而广泛应用。然而,当通过循环动态生成多个线性等式或不等式约束时,开发者可能会遇到约束不生效或结果不符合预期的情况。本文将详细解析导致此类问题的一个常见陷阱——Python中的“延迟绑定”(Late Binding),并介绍两种解决该问题的方法,最终引出并推荐使用scipy.optimize.LinearConstraint来更高效、准确地处理线性约束。
理解Python中的延迟绑定(Late Binding)
当在循环内部定义匿名函数(如lambda表达式)时,如果该函数引用了循环变量,那么这个变量的值通常会在函数实际被调用时才进行查找,而非在函数定义时立即绑定。这种行为被称为“延迟绑定”。
考虑以下示例:
numbers = [1, 2, 3]
funcs = []
for n in numbers:
funcs.append(lambda: n) # n在此处并未立即绑定
for func in funcs:
print(func())这段代码的输出将是:
3 3 3
而不是预期的1, 2, 3。这是因为当func()被调用时,循环已经完成,n的最终值是3,所有lambda函数都引用了同一个最终状态的n。
在scipy.optimize.minimize的约束定义中,如果像下面这样动态创建约束:
# 示例:错误的约束定义方式
cons = []
groups = [[0, 1, 2, 3], [4, 5], [6, 7, 8, 9]]
z_group = [0.25, 0.55, 0.2]
for idx, select in enumerate(groups):
# 此处的idx和select存在延迟绑定问题
cons.append({'type': 'eq', 'fun': lambda x: z_group[idx] - x[select].sum()})由于延迟绑定,所有生成的约束函数在执行时,idx和select都将是循环的最后一个值,导致只有最后一个组的约束生效。
解决延迟绑定问题
为了避免延迟绑定带来的问题,我们可以采用以下两种常见方法:
方法一:使用嵌套函数封装变量
通过定义一个外部函数,将循环变量作为参数传递给它,并在外部函数内部返回一个闭包(内部函数)。这样,循环变量会在外部函数被调用时立即绑定到内部函数的参数上,形成独立的上下文。
易标AI
告别低效手工,迎接AI标书新时代!3分钟智能生成,行业唯一具备查重功能,自动避雷废标项
135
查看详情
def create_group_constraint(idx, select_indices, target_value):
"""
创建一个用于特定组求和约束的内部函数。
idx: 组的索引
select_indices: 该组包含的变量索引
target_value: 该组变量之和的目标值
"""
def inner_constraint(x):
return target_value - x[select_indices].sum()
return inner_constraint
# 应用到约束列表中
cons = []
groups = [[0, 1, 2, 3], [4, 5], [6, 7, 8, 9]]
z_group = [0.25, 0.55, 0.2]
for idx, select in enumerate(groups):
cons.append({'type': 'eq', 'fun': create_group_constraint(idx, select, z_group[idx])})方法二:利用Lambda函数的默认参数
将循环变量作为lambda函数的默认参数传递。默认参数在函数定义时就会被评估和绑定,从而避免了延迟绑定。
cons = []
groups = [[0, 1, 2, 3], [4, 5], [6, 7, 8, 9]]
z_group = [0.25, 0.55, 0.2]
for idx, select in enumerate(groups):
# idx=idx 和 select=select 会在每次循环迭代时绑定当前值
cons.append({'type': 'eq', 'fun': lambda x, current_idx=idx, current_select=select: z_group[current_idx] - x[current_select].sum()})这两种方法都能有效解决延迟绑定问题,确保每个约束函数都引用了正确的idx和select值。
优化:利用Scipy的线性约束(LinearConstraint)
虽然上述方法解决了延迟绑定,但对于纯粹的线性约束,scipy.optimize提供了更高效、更健壮的LinearConstraint类。使用LinearConstraint有以下显著优势:
- 性能提升:优化算法能够识别线性约束的特殊结构,从而采用更专业的求解器和策略,显著提高收敛速度和效率。
- 数值稳定性:直接以矩阵形式定义线性约束,减少了浮点误差累积,提高了数值稳定性。
- 算法理解:优化器能够“理解”线性约束的几何特性(如可行域的边界),而不仅仅是判断当前点是否满足约束。
LinearConstraint的定义形式为:lb
让我们将总和约束和分组和约束转换为LinearConstraint的形式。
假设我们有10个变量x,目标函数opt_func和初始值x0:
import numpy as np
from scipy.optimize import minimize, LinearConstraint, Bounds
utility_vector = np.array([0.10, 0.08, 0.05, 0.075, 0.32,
0.21, 0.18, 0.05, 0.03, 0.12])
x0 = np.zeros((10,))
groups = [[0, 1, 2, 3], [4, 5], [6, 7, 8, 9]]
z_group = [0.25, 0.55, 0.2]
def opt_func(x, u, target):
utility = (x * u).sum()
return (utility - target)**2
n_variables = len(x0)
# 1. 定义总和约束:x.sum() = 1.0
# 对应 A @ x = 1.0,其中 A 是一个全为1的行向量
A_total_sum = np.ones((1, n_variables))
lb_total_sum = 1.0
ub_total_sum = 1.0
total_sum_constraint = LinearConstraint(A_total_sum, lb_total_sum, ub_total_sum)
# 2. 定义分组和约束:x[selection].sum() = Z_i
# 这需要构建一个矩阵 A_group_sum,其中每行对应一个组的约束
# 例如,对于 groups[0] = [0, 1, 2, 3],对应的行在索引0,1,2,3处为1,其余为0
A_group_sum = np.zeros((len(groups), n_variables))
lb_group_sum = np.array(z_group)
ub_group_sum = np.array(z_group)
for idx, select in enumerate(groups):
A_group_sum[idx, select] = 1
group_sum_constraint = LinearConstraint(A_group_sum, lb_group_sum, ub_group_sum)
# 3. 定义变量边界:x >= 0
# Scipy的minimize函数通常通过Bounds参数处理简单的变量边界
bounds = Bounds(0, np.inf, keep_feasible=True) # 所有x >= 0
# 4. 执行优化
res_linear = minimize(fun=opt_func,
x0=x0,
method='trust-constr', # 'trust-constr'方法支持LinearConstraint
bounds=bounds,
constraints=[total_sum_constraint, group_sum_constraint],
args=(utility_vector, 0.16),
tol=1e-4)
print(res_linear)
print(f'\nTotal allocation sum: {res_linear.x.sum():.4f}')
for idx, select in enumerate(groups):
print(f'Group {idx} ({select}) sum: {res_linear.x[select].sum():.4f}, target: {z_group[idx]}')
print(f' Difference: {z_group[idx] - res_linear.x[select].sum():.4e}')通过上述代码,我们可以看到LinearConstraint的强大之处。它将所有的线性约束集中表示,使得优化器能够更有效地利用这些信息。在实际应用中,这种方式通常能以更少的迭代次数达到更精确的解。
总结与最佳实践
在scipy.optimize.minimize中处理多重线性约束时,请牢记以下几点:
- 警惕延迟绑定:当在循环中动态创建lambda函数作为约束时,务必注意Python的延迟绑定机制。使用嵌套函数或lambda默认参数可以有效解决此问题。
- 优先使用LinearConstraint:对于任何可以表达为A @ x形式的线性等式或不等式约束,强烈建议使用scipy.optimize.LinearConstraint。这不仅能提升优化性能,还能增强数值稳定性和代码的可读性。
- 选择合适的优化方法:trust-constr是scipy.optimize.minimize中一个功能强大且推荐的算法,它能很好地处理各种类型的约束,包括LinearConstraint。
通过遵循这些最佳实践,您将能够更有效地构建和解决复杂的数值优化问题,确保约束的正确应用和优化的效率。
以上就是Scipy优化中多重线性约束的正确实现与性能优化的详细内容,更多请关注其它相关文章!
# 是一个
# 长沙小红书推广营销报价表
# 农村网站建设资金
# 武汉seo做排名
# 网站建设核心是什么
# 什么企业做网络营销推广
# 竞争对手网站优化
# 福建商务网站建设平台
# 深圳抖音seo优化招商
# 海阳个性化网站优化设计
# 济南网站建设报价表
# 迭代
# python
# 更有效地
# 的是
# 或不
# 如何使用
# 将是
# 会在
# 两种
# 绑定
# ai
# 工具
# app
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
网易大神怎么保存别人动态的图片_网易大神动态图片保存方法
JUnit5/Mockito:优雅测试内部依赖与异常处理的实践
b站赚钱渠道_b站收益来源
cad如何更改注释性对象的比例_cad注释性比例调整方法
冬*霸灯泡不亮怎么办_浴霸取暖灯一盏不亮的灯座清洁修复法
抖音极速版最新版本 抖音极速版官方下载地址
12306怎么选座位选到安静区_12306选座安静区域选择策略
AO3镜像入口大全 AO3网页版内容访问全集
C++如何检测键盘输入_C++ _kbhit与_getch函数非阻塞输入
Eclipse怎么运行工程_Eclipse工程运行配置说明
PHP URL参数传递与500错误调试指南
Win10怎么设置静态IP地址 Win10手动配置IP地址步骤【指南】
React/Next.js中实现列表项的动态选择与移动
Android Studio计算器C键功能异常排查与修复教程
HTML长属性值处理:表单action路径优化与代码规范应对
没有大陆身份证/银行卡如何实名微信? 亲测有效的几种方法分享
AO3中文官网链接_AO3网页版稳定镜像站
PHP表单数据传递:如何通过隐藏输入字段获取动态ID
win11 arm版怎么安装 M1/M2 Mac虚拟机安装ARM win11的方法
新三国志曹操传110级星符试炼夏侯渊极难攻略
如何在Promise链中有效终止错误处理后的执行
Node.js中HTML按钮与J*aScript函数交互的正确姿势
Mac怎么查看崩溃日志_Mac控制台错误报告分析
fishbowl官网免费版 fishbowl养鱼网站入口
优酷会员付费后没到账怎么办_优酷会员充值异常及解决方法
AO3最新镜像入口 Archive of Our Own官方平台访问
Win11怎么关闭触摸屏_Windows 11禁用HID符合标准触摸屏
AO3官方可用镜像 Archive of Our Own网页版最新入口
如何为你的Composer包编写自动化测试_集成PHPUnit到Composer的scripts工作流
J*a中实现Go语言select通道多路复用机制
格力空气能E5故障代码是什么情况_格力空气能E5代码解析与应对措施
蓝湖怎样用切图标注提对接效率_蓝湖用切图标注提对接效率【设计对接】
荣耀Play7TPro怎样在信息App置顶客服对话_iPhone荣耀Play7TPro信息App置顶客服对话【优先查看】
淘宝网网页版登录入口 淘宝官方网页版快捷登录
台积电1.4nm工艺A14瞄准2028:10年来性能提升80%
MAC怎么让Dock栏只显示当前运行的应用_MAC终端命令实现极简Dock栏
PHP中SSG-WSG API的AES加密实践:正确使用初始化向量
如何将HTML表格多行数据保存到Google Sheets
微信网页版登录教程_微信网页版登录入口在哪
高德地图公交到站提醒失败如何解决 高德提醒权限设置
怎样更改Windows系统的默认安装路径_避免C盘爆满的终极设置【技巧】
京东单号查询入口_京东快递订单追踪入口
XML中包含HTML标签导致解析错误? 正确嵌入非XML数据的两种方法
192.168.1.1管理中心入口 192.168.1.1路由器网页设置平台
抖音未来赚钱的新趋势 2025年值得关注的变现风口分析
支付宝如何管理隐私设置_支付宝隐私保护的配置技巧
J*aScript map 方法中处理循环元素为空数组的策略
Go语言中Map存储的结构体如何调用指针方法:深入解析与实践
Django表单验证失败时保留用户输入数据的最佳实践
Python异步编程实践:使用Binance API构建实时交易数据流


2025-11-08
浏览次数:次
返回列表