CPLEX杂记(五) 冲突检查

如果模型能够顺利求解,通过一组数据测试,通常是需要烧高香的事情。

更常发生的事情是模型解不了,然后我们需要挠破头皮去找哪些约束出了问题,还是输入数据出现异常,还是变量上下界不对,还是遇到了玄学。

CPLEX提供了冲突检查的功能,能够在某些情况下稍微帮助我们定位可能的问题所在。

这里我们通过一个简单小例子看一下冲突检测怎么整。

例子和建模

临近下班了,急着回家,这里就简单一点,直接拿CPLEX manual里的一个模型举例。

# 导入库
from docplex.mp.model import Model
from docplex.mp.model_reader import ModelReader
from collections import defaultdict
import os

# 直接从lp文件读取模型
lp_string = """
Minimize
 obj: cost
Subject To
 c1:  - cost + 80 x1 + 60 x2 + 55 x3 + 30 x4 + 25 x5 + 80 x6 + 60 x7 + 35 x8
      + 80 x9 + 55 x10  = 0
 c2:  x1 + x2 + 0.8 x3 + 0.6 x4 + 0.4 x5 >= 2.1
 c3:  x6 + 0.9 x7 + 0.5 x8 >= 1.2
 c4:  x9 + 0.9 x10 >= 0.8
 c5:  0.2 x2 + x3 + 0.5 x4 + 0.5 x5 + 0.2 x7 + 0.5 x8 + x10 - service  = 0
 c6:  x1 + x6 + x9 >= 1
 c7:  x1 + x2 + x3 + x6 + x7 + x9 >= 2
 c8:  x2 + x3 + x4 + x5 <= 0
 c9:  x4 + x5 + x8 <= 1
 c10: x1 + x10 <= 1
Bounds
 service >= 3.2
Binaries
 x1  x2  x3  x4  x5  x6  x7  x8  x9  x10
End
"""

# cplex没有提供直接读取string的接口,不得不进行个文件暂存操作
temp_input_file_name = "temp.lp"
with open(temp_input_file_name, "w") as f:
    f.write(lp_string)

mdl = ModelReader.read(temp_input_file_name, model_name = "InfeasibelLP")

# 清除临时文件
if os.path.exists(temp_input_file_name):
    os.remove(temp_input_file_name)

建完模型之后我们就可以尝试求解了,为了对比一下求解中输出的信息和冲突检测输出的信息,我们打印求解过程:

# 尝试求解
sol = mdl.solve(log_output=True)

得到的求解信息如下:

Version identifier: 12.10.0.0 | 2019-11-26 | 843d4de
CPXPARAM_Read_DataCheck                          1
Row 'c8' infeasible, all entries at implied bounds.
Presolve time = 0.00 sec. (0.00 ticks)

Root node processing (before b&c):
  Real time             =    0.00 sec. (0.01 ticks)
Parallel b&c, 16 threads:
  Real time             =    0.00 sec. (0.00 ticks)
  Sync time (average)   =    0.00 sec.
  Wait time (average)   =    0.00 sec.
                          ------------
Total (root+branch&cut) =    0.00 sec. (0.01 ticks)

求解器提示我们 c8不可行,我们可以打印所有约束出来看一眼:

print("----------------")
print("模型中共有{}条约束".format(mdl.number_of_constraints))
for ct in mdl.iter_constraints():
    print(ct)
----------------
模型中共有10条约束
c1: -cost+80x1+60x2+55x3+30x4+25x5+80x6+60x7+35x8+80x9+55x10 == 0
c2: x1+x2+0.800x3+0.600x4+0.400x5 >= 2.1
c3: x6+0.900x7+0.500x8 >= 1.2
c4: x9+0.900x10 >= 0.8
c5: 0.200x2+x3+0.500x4+0.500x5+0.200x7+0.500x8+x10-service == 0
c6: x1+x6+x9 >= 1.0
c7: x1+x2+x3+x6+x7+x9 >= 2.0
c8: x2+x3+x4+x5 <= 0
c9: x4+x5+x8 <= 1.0
c10: x1+x10 <= 1.0

于是我们看到了c8:x2+x3+x4+x5 <= 0,然后呢?为啥不可行??它和哪个其他约束冲突了???

这个模型比较小我们还可以手动试算一下,工业模型的规模能让人原地升天。

冲突检测

笔者没有在docplex模块中找到冲突检测的API,因此我们还是不得不借助cplex类提供的接口来进行冲突检测,示例代码如下:

# 获取cplex.Cplex()类对象
c = mdl.cplex
# 进行冲突检测
c.conflict.refine(c.conflict.all_constraints())
# 输出检测信息,再重新读入并在控制台中输出
# 需要吐槽的是Cplex并不支持以IOSteam为对象输入输出,因此不得不反复建立临时文件
output_fname = r"conflict.txt"
c.conflict.write(output_fname)
with open(output_fname, "r") as f:
    print(f.read())

得到如下的检测信息:

\ENCODING=ISO-8859-1
\Problem name: _conflict

Minimize
 obj1:
Subject To
 c2: x1 + x2 + 0.8 x3 + 0.6 x4 + 0.4 x5 >= 2.1
 c8: x2 + x3 + x4 + x5 <= 0
Bounds
 0 <= x1 <= 1
 0 <= x2 <= 1
 0 <= x3 <= 1
 0 <= x4 <= 1
 0 <= x5 <= 1
Binaries
 x1  x2  x3  x4  x5 
End

这里我们看到,是约束c2c8之间产生了冲突,我们的变量x1 - x5都是binary variable,因此要满足约束c8,只能让x2, x3, x4, x5都取0,而x1的上界是1,因此不能满足c2: x1>=2.1

急着回家就先这样吧,实际使用的时候通常还需要根据约束类型、变量类型、变量上下界等等对conflict groups进行筛选,再在各个组内进行检测,得到更加精细的冲突信息,读者可以自行研究。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容