2. 線性規(guī)劃:兩階段法python代碼

1. 補(bǔ)充問題

上一節(jié)中的代碼在運(yùn)行時(shí)還有很多細(xì)節(jié)沒有處理亚铁,這里補(bǔ)充兩個(gè)比較重要的情況:

  1. 存在等式約束
    如果有等式約束,那么就沒法通過添加松弛變量直接給出初始可行解,需要用大M法或者兩階段法求初始可行解。計(jì)算機(jī)求解一般使用二階段法偎行,即首先給等式約束條件添加人工變量,使得問題有一個(gè)初始可行解。注意人工變量和松弛變量/剩余變量不一樣蛤袒。松弛變量/剩余變量是用于將不等式變?yōu)榈仁剑?biāo)準(zhǔn)形)熄云,而人工變量則是在等式約束中額外添加一個(gè)變量,這個(gè)變量最后一定要等于0才行汗盘,否則就是沒有初始可行解皱碘,不能進(jìn)入第二階段。
    在第一階段隐孽,每個(gè)不等式約束對應(yīng)一個(gè)松弛變量癌椿,每個(gè)等式約束對應(yīng)一個(gè)人工變量。另外菱阵,b<0的不等式約束也需要添加人工變量踢俄,因?yàn)閷?yīng)的松弛變量等于b不能作為基變量。
  2. 沒有有限最優(yōu)解(目標(biāo)函數(shù)無界)
    對應(yīng)的情況是:目標(biāo)函數(shù)中有某個(gè)系數(shù)大于0晴及,對應(yīng)變量在約束條件中的系數(shù)全都≤0都办,問題結(jié)束,沒有最優(yōu)解虑稼。

2. python算法實(shí)現(xiàn)

首先假設(shè)問題除了不等式約束琳钉,還有等式約束:

求解問題為:min c*x
s.t.A_ub * x ≤ b_ub
    A_eq * x = b_eq

定義maxiter為最大迭代次數(shù),tol為求解精度蛛倦,OptimizeResult為格式輸出函數(shù)歌懒,_solve_simplex為上一節(jié)的求解函數(shù)。完整的求解步驟如下:

def linprog(c, A_ub=None, b_ub=None, A_eq=None, b_eq=None, maxiter=1000, tol=1.0E-12,):
    cc = np.asarray(c)
    f0 = 0
    n = len(c)
    mub = len(b_ub)
    meq = len(b_eq)
    m = mub+meq
    n_slack = mub
    n_artificial = meq + np.count_nonzero(bub < 0)
    Aub_rows, Aub_cols = A_ub.shape
    Aeq_rows, Aeq_cols = A_eq.shape
    slcount = 0
    avcount = 0
    status = 0
    messages = {0: "Optimization terminated successfully.",
                1: "Iteration limit reached.",
                2: "Optimization failed. Unable to find a feasible"
                   " starting point.",
                3: "Optimization failed. The problem appears to be unbounded.",
                4: "Optimization failed. Singular matrix encountered."}
    # 建立第一階段單純型表溯壶。
    T = np.zeros([m+2, n+n_slack+n_artificial+1])
    T[-2, :n] = cc
    T[-2, -1] = f0
    b = T[:-2, -1]
    if meq > 0:
        T[:meq, :n] = Aeq
        b[:meq] = beq
    if mub > 0:
        T[meq:meq+mub, :n] = Aub
        b[meq:meq+mub] = bub
        np.fill_diagonal(T[meq:m, n:n+n_slack], 1)

    #第一階段的基向量basis及皂,記錄下標(biāo)。
    basis = np.zeros(m, dtype=int)
    r_artificial = np.zeros(n_artificial, dtype=int)
    for i in range(m):
        if i < meq or b[i] < 0:
            basis[i] = n+n_slack+avcount
            r_artificial[avcount] = i
            avcount += 1
            if b[i] < 0:
                b[i] *= -1
                T[i, :-1] *= -1
            T[i, basis[i]] = 1
            T[-1, basis[i]] = 1
        else:
            basis[i] = n+slcount
            slcount += 1

    for r in r_artificial:
        T[-1, :] = T[-1, :] - T[r, :]
  
    #第一階段求解
    nit1, status = _solve_simplex(T, n, basis)

    # 如果第一階段的目標(biāo)函數(shù)為0且改,則刪除人工變量验烧,進(jìn)入第二階段
    if abs(T[-1, -1]) <= sol:
        T = T[:-1, :]
        T = np.delete(T, np.s_[n+n_slack:n+n_slack+n_artificial], 1)
    else:
        status = 2

    if status != 0:
        message = messages[status]
        if disp:
            print(message)
        return OptimizeResult(x=np.nan, fun=-T[-1, -1], nit=nit1, status=status, message=message, success=False)
    nit2, status = _solve_simplex(T, n, basis, maxiter=maxiter-nit1, phase=2, callback=callback, tol=tol, nit0=nit1)

    solution = np.zeros(n+n_slack+n_artificial)
    solution[basis[:m]] = T[:m, -1]
    x = solution[:n]
    slack = solution[n:n+n_slack]
    obj = -T[-1, -1]
    return OptimizeResult(x=x, fun=obj, nit=int(nit2), status=status, slack=slack, message=messages[status], success=(status == 0))

下面是例子:

import numpy as np
c=np.array([2,3,-5])
A_ub=np.array([[-2,5,-1],[1,3,1]])
B_ub=np.array([-10,12])
A_eq=np.array([[1,1,1]])
B_eq=np.array([7])
res=linprog(-c,A_ub,B_ub,A_eq,B_eq)
print(res)

輸出為

     fun: -14.571428571428571
 message: 'Optimization terminated successfully.'
     nit: 2
   slack: array([3.85714286, 0.57142857, 6.42857143, 7.        , 0.        ])
  status: 0
 success: True
       x: array([6.42857143, 0.57142857, 0.        ])
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市又跛,隨后出現(xiàn)的幾起案子碍拆,更是在濱河造成了極大的恐慌,老刑警劉巖慨蓝,帶你破解...
    沈念sama閱讀 211,265評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件感混,死亡現(xiàn)場離奇詭異,居然都是意外死亡菌仁,警方通過查閱死者的電腦和手機(jī)浩习,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,078評論 2 385
  • 文/潘曉璐 我一進(jìn)店門静暂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來济丘,“玉大人,你說我怎么就攤上這事∧∶裕” “怎么了疟赊?”我有些...
    開封第一講書人閱讀 156,852評論 0 347
  • 文/不壞的土叔 我叫張陵,是天一觀的道長峡碉。 經(jīng)常有香客問我近哟,道長,這世上最難降的妖魔是什么鲫寄? 我笑而不...
    開封第一講書人閱讀 56,408評論 1 283
  • 正文 為了忘掉前任吉执,我火速辦了婚禮,結(jié)果婚禮上地来,老公的妹妹穿的比我還像新娘戳玫。我一直安慰自己,他們只是感情好未斑,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,445評論 5 384
  • 文/花漫 我一把揭開白布咕宿。 她就那樣靜靜地躺著,像睡著了一般蜡秽。 火紅的嫁衣襯著肌膚如雪府阀。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,772評論 1 290
  • 那天芽突,我揣著相機(jī)與錄音试浙,去河邊找鬼。 笑死诉瓦,一個(gè)胖子當(dāng)著我的面吹牛川队,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播睬澡,決...
    沈念sama閱讀 38,921評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼固额,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了煞聪?” 一聲冷哼從身側(cè)響起斗躏,我...
    開封第一講書人閱讀 37,688評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎昔脯,沒想到半個(gè)月后啄糙,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,130評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡云稚,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,467評論 2 325
  • 正文 我和宋清朗相戀三年隧饼,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片静陈。...
    茶點(diǎn)故事閱讀 38,617評論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡燕雁,死狀恐怖诞丽,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情拐格,我是刑警寧澤僧免,帶...
    沈念sama閱讀 34,276評論 4 329
  • 正文 年R本政府宣布,位于F島的核電站捏浊,受9級特大地震影響懂衩,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜金踪,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,882評論 3 312
  • 文/蒙蒙 一浊洞、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧胡岔,春花似錦沛申、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,740評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至奕锌,卻和暖如春著觉,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背惊暴。 一陣腳步聲響...
    開封第一講書人閱讀 31,967評論 1 265
  • 我被黑心中介騙來泰國打工饼丘, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人辽话。 一個(gè)月前我還...
    沈念sama閱讀 46,315評論 2 360
  • 正文 我出身青樓肄鸽,卻偏偏與公主長得像,于是被迫代替她去往敵國和親油啤。 傳聞我的和親對象是個(gè)殘疾皇子典徘,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,486評論 2 348

推薦閱讀更多精彩內(nèi)容

  • 本章涉及知識點(diǎn)1、線性規(guī)劃的定義2益咬、可行區(qū)域逮诲、目標(biāo)函數(shù)、可行解和最優(yōu)解3幽告、轉(zhuǎn)線性規(guī)劃為標(biāo)準(zhǔn)型4梅鹦、轉(zhuǎn)線性規(guī)劃為松弛型...
    PrivateEye_zzy閱讀 38,428評論 2 36
  • 轉(zhuǎn)載請注明出處 http://www.reibang.com/p/dd0761a2fdfd作者:@貳拾貳畫生 單純...
    貳拾貳畫生閱讀 50,641評論 10 38
  • ?一般來說凸優(yōu)化(Convex Optimization, CO)中最一般的是錐規(guī)劃 (Cone Programm...
    史春奇閱讀 5,057評論 1 6
  • 今天在家整理全域旅游的講課資料。 全域旅游冗锁,概念沒記住齐唆,記住了某省副省長的八個(gè)字,處處冻河,時(shí)時(shí)箍邮,行行抛腕,人人。處處是景...
    西瓜貓貓閱讀 168評論 0 0
  • 一媒殉、什么是Handler Handler是android給我們提供用來更新UI的機(jī)制,同時(shí)也是消息處理機(jī)制摔敛,可以通...
    瓦西大人閱讀 213評論 1 3