LINGO大规模规划求解
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
lingo 大规模规划求解
首先,让我们先看看一个非常简单的规划例子在LINGO软件中实现过程:
目标函数:
约束条件:
求解上面目标函数的最小值,我们在lingo中可编写如下代码:model:
MIN=2*X1+X2-3*X3+5;
X1+X2-3*X3<=10;
X1-2*X2>=5
@GIN(X1);!整数约束;
@GIN(X2);
@GIN(X3);
END
可以看出,LINGO语言和数学专业语言很接近,很容易表示约束条件和目标函数。
可是对于规模很大的约束条件,难道我们也必须这样一条一条的输入吗,显然这样做是一件非常困难和繁琐的事,lingo语言又是如何表示约束条件规模巨大的规划问题呢,带着这样的疑问,让我们一步一步得看下面的内容:
一、集合域
在数学中集合的定义如下:
集合:具有某种相同属性的对象放在一起,就形成了一个集合,
集合中的每一个对象称作该集合的元素。
在大规模的优化问题中,集合是必然存在的,比如在平板车建模中,各种规格集装箱就可以看成一个集合,在货物配送问题中154个城市可以看成一个集合。
在lingo语言中,将某些对象看成一个集合便可以很方便地对集合中的每一个元素进行统一处理。
集合域必须在模型的约束引用集合之前定义。
集合域用关键字“sets”开始,“endsets”结束。
集合分类:
基本集合定义统一语法格式:
setname[/member-list/][:attribute-list];
集合名/对象名1 对象名2 … 对象名n/:对象属性;
集合定义的几种方法:
sets
row/1..20/:d1,d2,…dn; !集合名/对象名/:对象属性;end sets
派生集合定义方法:
sets
row/1..20/;
col/1..100/;
page/1..50/;
link(row.col):k1,k2…kn;
trd/(row.col.page):t1,t2,…tn;
end sets
K1可以表示某个省的某个城市的人口。
t1可以表示某个省的某个城市某个人的收入。
二、数据域:
数据域是优化问题中已知得对象的属性值,例如:人的身高,体重;车辆的载重,行驶速度。
数据域以关键字“data”开头,“enddata”结束。
数据域可以出现在模型中的任何地方。
data:
d1=…;
k1=…;
t1=…;
enddata
数据域的未知数值
有时只想为一个集的部分成员的某个属性指定值,而让其余成员的该属性保持未知,以便让LINGO去求出它们的最优值。在数据声明中输入两个相连的逗号表示该位置对应的集成员的属性值未知。两个逗号间可以有空格。
例3.8
sets:
years/1..5/: capacity;
endsets
data:
capacity = ,34,20,,;
enddata
属性capacity的第2个和第3个值分别为34和20,其余的未知。
实时数据处理
在某些情况,对于模型中的某些数据并不是定值。譬如模型中有一个通货膨胀率的参数,我们想在2%至6%范围内,对不同的值求解模型,来观察模型的结果对通货膨胀的依赖有多么敏感。我们把这种情况称为实时数据处理(what if analysis)。LINGO有一个特征可方便地做到这件事。
在本该放数的地方输入一个问号(?)。
data:
interest_rate,inflation_rate = .085 ?;
enddata
每一次求解模型时,LINGO都会提示为参数inflation_rate输入一个值。在WINDOWS操作系统下,将会接收到一个类似下面的对话框:
直接输入一个值再点击OK按钮,LINGO就会把输入的值指定给inflation_rate,然后继续求解模型。
除了参数之外,也可以实时输入集的属性值,但不允许实时输入集成员名。
三、初始域
初始域是专门为初始化决策变量而定义的一块区域。
初始部分是LINGO提供的另一个可选部分。在初始部分中,可以输入初始声明(initialization statement),和数据部分中的数据声明相同。对实际问题的建模时,初始部分并不起到描述模型的作用,在初始部分输入的值仅被LINGO求解器当作初始点来用,并且仅仅对非线性模型有用。和数据部分指定变量的值不同,LINGO求解器可以自由改变初始部分初始化的变量的值。
一个初始部分以“init:”开始,以“endinit”结束。初始部分的初始声明规则和数据部分的数据声明规则相同。也就是说,我们可以在声明的左边同时初始化多个集属性,可以把集属性初始化为一个值,可以用问号实现实时数据处理,还可以用逗号指定未知数值。
init:
X, Y = 0, .1;
endinit
Y=@log(X);
X^2+Y^2<=1;
好的初始点会减少模型的求解时间。
四、集合循环函数
统一语法格式:
@function(setname[(set_index_list)[| condition]]:expression_list);
其中:
function是集合函数名,是FOR,MAX,MIN,PROD,SUM五种之一;setname是集合名;
set_index_list是集合索引列表(不需使用索引时可以省略);condition是逻辑表达式描述的过虑条件(通常含有索引,无条件时可省略);
expression_list是一个表达式(对@FOR函数,可以是一组表达式)。五个集合函数名的含义如下:
@FOR(集合元素循环函数):对集合setname 的每个元素独立地生成表达式,表达式
由expression_list描述(通常是优化问题的约束)。
@MAX(集合属性的最大值函数):返回集合setname上的表达式的最大值。
@MIN(集合属性的最小值函数):返回集合setname上的表达式的最小值。
@PROD(集合属性的乘积函数):返回集合setname上的表达式的积。@SUM(集合属性的求和函数):返回集合setname上的表达式的和。对应数学表达式与LINGO命令:
数学表达式lingo命令
(一维数组和)@sum(row(i):x(i));
(二维维数组和)@sum(links(i,j):x(i,j));
(矩阵点乘和)@sum(links(i,j):x(i,j)*Y(i,j)); (求数组的积)@prod(row(i):x(i));
(比较m个一维数组的和)@for(col(j):@sum(row(i):x(i))
<=
@sum(row(i):y(i)));
五、变量定界函数