首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >问答首页 >python中的约束优化,其中一个变量依赖于另一个变量

python中的约束优化,其中一个变量依赖于另一个变量
EN

Stack Overflow用户
提问于 2019-05-08 14:12:23
回答 1查看 1.1K关注 0票数 1

我有一个问题,我有4个变量x1, x2, x3 and x4。我需要在满足以下条件的情况下找到x1, x2, x3, x4的值:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1. 1995 < 2*x1 + 4*x2 + 3*x3 + x4 < 2000
2. x1 >= 1.2*x2
3. x2 >= 1.3*x3
4. x3 >= 1.1*x4
5. x4 > 0.0

我可以使用python-constraint (https://labix.org/python-constraint)来解决这个问题,但是在我的系统上需要大约30分钟才能解决这个问题,这太长了。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
from constraint import *

problem = Problem()
problem.addVariable("x1", range(100,500))
problem.addVariable("x2", range(100,500))
problem.addVariable("x3", range(100,500))
problem.addVariable("x4", range(100,500))

problem.addConstraint(lambda a, b, c, d: 2*a + 3*b + 4*c + 5*d > 1995, ["x1", "x2", "x3", "x4"])
problem.addConstraint(lambda a, b, c, d: 2*a + 3*b + 4*c + 5*d < 2005, ["x1", "x2", "x3", "x4"])
problem.addConstraint(lambda a, b: a >= 1.2 * b, ["x1", "x2"])
problem.addConstraint(lambda b, c: b >= 1.3 * c, ["x2", "x3"])
problem.addConstraint(lambda c, d: c >= 1.1 * d, ["x3", "x4"])
problem.addConstraint(lambda d: d > 0, ["x4"])

problem.getSolutions()

我还查看了scipy.optimize.linprog,但我找不到传递条件2、3和4的方法,因为它依赖于同一问题中另一个变量的值。我可以使用bounds参数传递每个单独变量的边界,如下所示:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
x1_bounds = (100, 200)
x2_bounds = (200, 300)

但是如何在界限中传递其他变量的值,比如x1_bounds >= 1.2*x2?或者有没有其他方法可以做到这一点?

这个问题可以用excel中的GRG非线性求解器来解决,但是我在python中找不到等价物。

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2019-05-08 15:49:40

你的问题实际上是线性的,所以它非常适合线性规划方法。然而,你把它交给求解器,却没有关于问题线性的线索,所以它一定会发现这很棘手:它几乎必须尝试每一种可能性,这将需要很长时间。可以将约束重写为python-constraint求解器的不同形式(例如,它具有MaxSumConstraint约束形式),这可能会更好地工作,但理想情况下,我认为您应该使用专门用于线性问题的求解器。

有一个名为kiwisolver的求解器,它会做你想做的事情。下面是为该库转换的示例:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import kiwisolver

x1 = kiwisolver.Variable('x1')
x2 = kiwisolver.Variable('x2')
x3 = kiwisolver.Variable('x3')
x4 = kiwisolver.Variable('x4')

constraints = [1995 <= 2*x1 + 4*x2 + 3*x3 + x4,
               2*x1 + 4*x2 + 3*x3 + x4 <= 2000,
               x1 >= 1.2*x2,
               x2 >= 1.3*x3,
               x3 >= 1.1*x4,
               x4 >= 0]

solver = kiwisolver.Solver()

for cn in constraints:
    solver.addConstraint(cn)

for x in [x1, x2, x3, x4]:
    print(x.value())

这给了我们

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
254.49152542372883
212.07627118644066
163.13559322033896
148.30508474576254

但您也可以使用标准的线性规划求解器,如scipy。你只需要将你的不平等重新组织成正确的形式。

您需要:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1. 1995 < 2*x1 + 4*x2 + 3*x3 + x4 < 2000
2. x1 >= 1.2*x2
3. x2 >= 1.3*x3
4. x3 >= 1.1*x4
5. x4 > 0.0

因此,我们将其重写为:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 2*x1 +  4*x2 +  3*x3 +  1*x4 < 2000
-2*x1 + -4*x2 + -3*x3 + -1*x4 < -1995
-1*x1 + 1.2*x2 + 0*x3 +  0*x4 < 0
 0*x1 + -1*x2 + 1.3*x3 + 0*x4 < 0
 0*x1 +  0*x2 + -1*x3 + 1.1*x4 < 0

您可以像您在问题中提到的那样,向x4添加x1的界限,但默认情况下,它们将是非负的。因此,对于LP,我们还需要选择在可能的解的多面体中的什么位置进行优化:在这种情况下,我将选择具有最小和的解。这就给了我们这个:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
from scipy.optimize import linprog

output = linprog([1, 1, 1, 1],
                [[ 2,   4,   3,   1],
                 [-2,  -4,  -3,  -1],
                 [-1, 1.2,   0,   0],
                 [0,   -1, 1.3,   0],
                 [0,    0,  -1, 1.1]],
                [2000, -1995, 0, 0, 0])

print(output.x)

这给了我们

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
[274.92932862 229.10777385 176.23674912   0.        ]

这是最优的LP解决方案。注意,它使得x4 = 0:LP通常不区分>>=,因此我们有一个解决方案,其中x4是零,而不是一个大于零的小epsilon。

最后,注意这个问题是强欠约束的:我们可以通过改变目标来选择一个完全不同的解决方案。这是一个让linprog最大化2*x1 + 4*x2 + 3*x3 + x4的解决方案

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
from scipy.optimize import linprog


output = linprog([-2, -4, -3, -1],
                 [[ 2,   4,   3,  1],
                  [-2,  -4,  -3, -1],
                  [-1, 1.2,   0, 0],
                  [0,   -1, 1.3, 0],
                  [0,    0,  -1, 1.1]],
                 [2000, -1995, 0, 0, 0])

print(output.x)

给予

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
[255.1293488  212.60779066 163.54445436 148.67677669]
票数 2
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/56042803

复制
相关文章
python中的变量
  什么是变量?总结不好也记不得它的完整定义,就举个例子来便于自己学习总结吧。假如我们要计算1+2的值,那么首先在内存中要存储两个值,一个是:1,一个是:2。内存中有了这两个值,计算的时候就会找到它们进行计算。但是我们可能是计算任意两个数相加,数学中我们常用 x+y 来表示两个任意的数相加。假如在程序中我们用a+b来表示两个数相加,那么当a=1,b=2时,就可以计算出1+2=3,此时这个a和b就是变量,它们也可以等于其他数值,结果也是随着数值的改变而改变的。a和b的值能变动,就叫变量。刚才说了内存中存了1和2,那么a和b与内存中的1和2有什么联系呢,可以理解为a=1,b=2就是把内存中的1取个名字叫a,内存中的2取个名字叫b。这样按名字就能找到它的值了。
py3study
2020/01/19
2.5K0
python中的变量
python中的变量
本文介绍了Python中的变量和运算符,包括整数、浮点数、布尔数、字符串和列表等数据类型的操作和应用。同时,还介绍了Python中的字符串格式化、运算符的优先级和字符串的切片等知识点。
企鹅号小编
2017/12/28
2.8K0
python中的变量
Python中类变量、成员变量、局部变量的区别
class A:v1 = 100 # 类变量def __init__(self):self.v2 = 200 # 成员变量v3 = 300 # 局部变量类变量可以由类名统一修改:A.v1 = 300# 则每一个A实例里v1都变成300成员变量只能由实例自己改变:A.v2 # 这个是错的。a = A()a.v2 # 这个是可以访问的。局部变量只在函数内部生效
狼啸风云
2020/10/16
4.4K0
python中的变量
通过上面的对比可以看到,python2中在运行赋值运算符的时候,变量始终是整型,而在python3中,变量在做除法运算符的时候会变为浮点型。
嘻哈记
2021/03/20
2K0
在JSP页面中调用另一个JSP页面中的变量
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/huyuyang6688/article/details/16896447
DannyHoo
2018/09/13
7.8K0
java定义全局变量的方法_java调用另一个类的变量
1、单独写一个final的类,在里面定义final static的全局变量,在其它程序里包含进来就可以了。
全栈程序员站长
2022/09/24
2.6K0
python开发_python中的变量:全局变量和局部变量
如果你在为python中的变量:全局变量和局部变量头疼,我想这篇blog会给你帮助
Hongten
2018/09/13
1.1K0
python开发_python中的变量:全局变量和局部变量
【PYTHON】论Python的类中的私有变量、共有变量、全局变量、局部变量
Python语言中,一切皆对象。对于Python语言中的变量,我想大家都不陌生,但是在Python中,变量也有很多种类。如私有变量、共有变量、全局变量、局部变量等。这里给大家介绍Python类(class)中的变量之间的区别。参考代码:
用户7886150
2020/11/26
4.4K0
干货 | Python中的变量
Python中的变量是什么呢?变量,记录事物变化的状态。是为了让计算机具备人的某项功能,能更好的完成人类下达的任务,由此诞生了变量。
码农向前冲
2022/01/06
9440
干货 | Python中的变量
9 python 中的变量
如果你要实现从字符串中获取一段子字符串的话,可以使用变量 [头下标:尾下标],就可以截取相应的字符串,其中下标是从 0 开始算起,可以是正数或负数,下标可以为空表示取到头或尾。
py3study
2020/01/06
9450
关于python中全局变量和局部变量的一个问题
众所周知,全局变量和局部变量可以说是泾渭分明,如 a = 10 def test(): a = 20 print(a) test() #将输出局部变量a print(a) # 将输出全局变量a #答案是: 20 10 函数内部的局部变量a,并不会影响全局变量a 但是假设为列表或字典呢? my_list = [10] def list_test(): my_list=[] my_list.append(20) print(‘函数的’,my_list) list_test() print(my_list) 答案为:函数的 [20] [10] 仍然符合局部变量并不会影响全局变量 注意:假设为一个空的列表或字典 my_list = [] def list_test(): my_list.append(20) print(‘函数的’,my_list) list_test() print(my_list) 答案:函数的 [20] [20] 这里为何是20,而不是[]? 个人见解:函数内部无my_list这个列表,因此他将会调用全局变量my_list列表,然后把元素放在了列表中,改变的是全局变量,又或是因为是容器? 望大佬们积极解惑,不胜感激
用户7886150
2021/01/20
6690
python中变量命名
*from module import *时候,仅仅会导入__all__中定义的变量。
全栈程序员站长
2022/07/08
5990
python中变量命名
python中全局变量、局部变量、类变量、实例变量简析
因为python为动态语言,处理变量的方式与一些静态语言(比如C++)不大一样,在这里对这些变量进行小小的总结
用户7886150
2020/11/23
1.8K0
类其中的变量为final时的用法
类当中final变量没有初始缺省值,必须在构造函数中赋值或直接当时赋值。否则报错。
马克java社区
2021/03/15
4510
Python 中的变量作用域
通常,在块中定义的变量仅在该块中可用。在街区外无法到达。这样的变量称为局部变量。形式参数标识符也表现为局部变量。
很酷的站长
2023/08/15
1740
Python 中的变量作用域
python中的变量命名规则
There are only two hard things in Computer Science: cache invalidation and naming things. 在计算机科学领域只有两件难事:缓存失效 和 给东西起名字
全栈程序员站长
2022/09/02
1.2K0
Python中变量的作用域
我们在写代码过程中会用到很多变量,这些变量会出现在各种代码块中,有的出现在函数块里,有的在函数块外,例如:
py3study
2020/01/19
1.1K0
python 中变量和对象
在 python 中,类型属于对象,变量是没有类型的: a=[1,2,3] a="Runoob" 以上代码中,[1,2,3] 是 List 类型,"Runoob" 是 String 类型,而变量 a 是没有类型,她仅仅是一个对象的引用(一个指针),可以是 List 类型对象,也可以指向 String 类型对象。 可更改(mutable)与不可更改(immutable)对象 在 python 中,strings, tuples, 和 numbers 是不可更改的对象,而 list,dict 等则是可以修改的对象。 不可变类型:变量赋值 a=5 后再赋值 a=10,这里实际是新生成一个 int 值对象 10,再让 a 指向它,而 5 被丢弃,不是改变a的值,相当于新生成了a。 可变类型:变量赋值 la=[1,2,3,4] 后再赋值 la[2]=5 则是将 list la 的第三个元素值更改,本身la没有动,只是其内部的一部分值被修改了。 python 函数的参数传递: 不可变类型:类似 c++ 的值传递,如 整数、字符串、元组。如fun(a),传递的只是a的值,没有影响a对象本身。比如在 fun(a)内部修改 a 的值,只是修改另一个复制的对象,不会影响 a 本身。 可变类型:类似 c++ 的引用传递,如 列表,字典。如 fun(la),则是将 la 真正的传过去,修改后fun外部的la也会受影响 python 中一切都是对象,严格意义我们不能说值传递还是引用传递,我们应该说传不可变对象和传可变对象。
py3study
2020/01/15
9680
python中全局变量和局部变量详解
python中全局变量和局部变量的最大区别在于局部变量只能通过函数去访问,而全局变量可以直接访问。
狼啸风云
2022/11/18
4440
点击加载更多

相似问题

如何在优化算法中引入依赖于决策变量的计算变量的线性约束

10

基于python的多约束多变量优化

111

python中的非线性约束优化--变量和除法

173

如果变量依赖于优化变量,如何编写优化程序?

10

Matlab优化-约束优化变量的条件和

20
添加站长 进交流群

领取专属 10元无门槛券

AI混元助手 在线答疑

扫码加入开发者社群
关注 腾讯云开发者公众号

洞察 腾讯核心技术

剖析业界实践案例

扫码关注腾讯云开发者公众号
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文