对偶单纯形法算法精要

news/2025/1/21 18:50:10/文章来源:https://www.cnblogs.com/haohai9309/p/18390966

单纯形法是线性规划中最经典且广泛应用的求解方法,通过在可行解的边界上移动,逐步逼近最优解。它从一个初始基本可行解开始,不断优化目标函数值,直到找到最优解。对偶单纯形法则是单纯形法的一种变形,尤其适用于特定类型的线性规划问题。不同于标准的单纯形法,对偶单纯形法从一个对偶可行但原始不可行的初始解出发,通过逐步改善解的可行性和最优性,最终找到最优解。对偶单纯形法在快速调整和重新优化方面表现出色,特别是在处理新的约束或变量被添加到现有模型中的情况时,这使得它在实际应用中具有独特的优势。

单纯形法比较 对偶单纯形流程图

一、对偶单纯形算法

‌对偶单纯形法是一种用于求解线性规划问题的数学方法,由美国数学家‌C.莱姆基在1954年提出。这种方法基于‌对偶理论,通过迭代过程逐步搜索原始问题的最优解。与单纯形法不同,对偶单纯形法从满足对偶可行性条件出发,通过迭代逐步逼近原始问题的最优解。在迭代过程中,始终保持基解的对偶可行性,使不可行性逐步消失。‌
对偶单纯形法的基本思想是:从原规划的一个对偶可行解出发;然后检验原规划的基本解是否可行,即是否有负的分量,如果有小于零的分量,则进行迭代,求另一个基本解,此基本解对应着另一个对偶可行解(检验数非正)。如果得到的基本解的分量皆非负,则该基本解为最优解。也就是说,对偶单纯形法在迭代过程中始终保持对偶解的可行性(即检验数非正),使原规划的基本解由不可行逐步变为可行,当同时得到对偶规划与原规划的可行解时,便得到原规划的最优解。

1.1 对偶问题算法步骤

a)根据线性规划典式形式,建立初始对偶单纯形表。此表对应原规划的一个基本解。要求检验数行各元素一定非正,原规划的基本解允许有小于零的分量;
b)若基本解的所有分量皆非负,则得到原规划的最优解,停止计算;若基本解中有小于零的分量\(b_j < 0\),并且\(b_j\)所在行各系数\(a_{ij} \geq 0\),则原规划没有可行解,停止计算;若\(b_j < 0\),并且存在\(a_{ij} < 0\),则确定\(x_i\)为出基变量,并计算

\[\theta = \min \left\{ \frac{\sigma_j}{a_{ij}} \middle| a_{ij} < 0 \right\} = \frac{\sigma_k}{a_{ik}} \]

确定\(x_k\)为进基变量。若有多个\(b_j < 0\),则选择最小的进行分析计算;
c)以\(a_{ik}\)为中心元素,按照与单纯形类似的方法,在表中进行迭代计算,返回步骤 b)。

1.2 练习

例1:用对偶单纯形法求解下面线性规划

\[\begin{aligned} \text{min}\quad w= 2x_1 + 3x_2 + 4x_3 \\ \text{s.t. } \begin{cases} x_1 + 2x_2 + x_3 \geq 3 \\ 2x_1 - x_2 + 3x_3 \geq 4 \\ x_1, x_2, x_3 \geq 0 \end{cases} \end{aligned} \]

\[\begin{aligned} \text{max } \quad w= -2x_1 - 3x_2 - 4x_3 \\ \text{s.t. } \begin{cases} -x_1 - 2x_2 - x_3 + x_4 = -3 \\ -2x_1 + x_2 - 3x_3 + x_5 = -4 \\ x_1, \ldots, x_5 \geq 0 \end{cases} \end{aligned} \]

得原问题的最优解为$x_1 = 11/5,x_2 = 2/5,x_3 = 3$;最优值为$w = 28/5$。

二、例题

利用对偶单纯形法求解线性规划问题:

\[\begin{cases} \min f(x) = 4x_{1} + 12x_{2} + 18x_{3} \\ \text{s.t. } x_{1} + 3x_{3} \geq 3 \\ \quad \quad 2x_{2} + 2x_{3} \geq 5 \\ \quad \quad x_{1}, x_{2}, x_{3} \geq 0 \end{cases} \]

解:先标准化:

\[\begin{cases} \max f(x) =- 4x_{1} - 12x_{2} - 18x_{3} \\ \text{s.t.}\quad x_{1} + 3x_{3} - x_{4} = 3 \\ \quad \quad 2x_{2} + 2x_{3} - x_{5} = 5 \\ \quad \quad x_{1}, x_{2}, x_{3}, x_{4}, x_{5} \geq 0 \end{cases} \]

为了将 $$ B = (p_{4}, p_{5}) $$ 作为初始基,对约束条件两边同乘 \(-1\)

\[\begin{cases} -x_{1} - 3x_{3} + x_{4} = -3 \\ -2x_{2} - 2x_{3} + x_{5} = -5 \end{cases} \]

\[C^{T}-c_{B}^{T}BA = (-4, -12, -18, 0, 0)^{T} \]

对应的初始单纯形表如下:

变量 $$ x_{1} $$ $$ x_{2} $$ $$ x_{3} $$ $$ x_{4} $$ $$ x_{5} $$
$$ f $$ 0 -4 -12 -18 0 0
$$ x_{4} $$ -3 -1 0 -3 1 0
$$ x_{5} $$ -5 0 -2 -2 0 1

当前解 $$ x = (0, 0, 0, -3, -5)^{T} $$ 为非可行解,但是它是对偶可行解。

因为:

\[\min\left\{ -3, -5 \right\} = -5, \quad r = 2, \quad x_{j2} = x_{5} \text{ 为离基变量} \]

\[\frac{b_{0s}}{b_{rs}} = \min\left\{ \frac{b_{0j}}{b_{rj}} \mid b_{rj} < 0 \right\} = \min\left\{ \frac{-12}{-2}, \frac{-18}{-2} \right\} = 6 = \frac{b_{02}}{b_{22}}, \quad s = 2 \]

因此,进基变量 $$ x_{s} = x_{2} $$,且 $$ b_{22} = -2 $$ 为主元。

经旋转变换后,有:

变量 $$ x_{1} $$ $$ x_{2} $$ $$ x_{3} $$ $$ x_{4} $$ $$ x_{5} $$
$$ f $$ 30 -4 0 -6 0 -6
$$ x_{4} $$ -3 -1 0 -3 1 0
$$ x_{2} $$ 5/2 0 1 1 0 -1/2

此时 $$ b_{i0} \geq 0 , (i=1,...,m) $$ 不成立,仍为不可行解。

因为:

\[b_{r0} = \min\left\{b_{i0} \mid b_{i0} < 0, i=1,...,m \right\} = \min\left\{ -3\right\} = -3, \quad x_{4} \text{ 为离基变量}, \quad r = 1 \]

\[\frac{b_{0s}}{b_{rs}} = \min\left\{ \frac{b_{0j}}{b_{rj}} \mid b_{rj} < 0 \right\} = \min\left\{ \frac{-4}{-1}, \frac{-6}{-3} \right\} = 2 = \frac{b_{03}}{b_{r3}}, \quad s = 3 \]

因此,进基变量 $$ x_{s} = x_{3} $$,且 $$ b_{13} = -3 $$ 为主元。

经旋转变换后,有:

变量 $$ x_{1} $$ $$ x_{2} $$ $$ x_{3} $$ $$ x_{4} $$ $$ x_{5} $$
$$ f $$ 36 -2 0 0 -2 -6
$$ x_{3} $$ 1 1/3 0 1 -1/3 0
$$ x_{2} $$ 3/2 -1/3 1 0 1/3 -1/2

此时 $$ b_{i0} \geq 0, (i=1,...,m) $$ 成立,是可行解,算法终止。

最优解和最优值:

\[\bar{x} = \left(0, \frac{3}{2}, 1\right)^{T} \quad f(\bar{x}) = 36 \]

三、对偶单纯形的Python求解

假设食物的各种营养成分、价格如下表:

Food Energy (能量) Protein (蛋白质) Calcium (钙) Price
Oatmeal (燕麦) 110 4 2 3
Whole milk (全奶) 160 8 285 9
Cherry pie (樱桃派) 420 4 22 20
Pork with beans (猪肉) 260 14 80 19

要求我们买的食物中,至少要有2000的能量,55的蛋白质,800的钙,怎样买最省钱?

设买燕麦、全奶、樱桃派、猪肉为\(x_1, x_2, x_3, x_4\),于是可以写出如下的不等式组:

\[\begin{aligned} & \min \quad 3x_1 + 9x_2 + 20x_3 + 19x_4 \quad \text{money} \\ & \text{s.t.} \\ & 110x_1 + 160x_2 + 420x_3 + 260x_4 \geq 2000 \quad \text{energy} \\ & 4x_1 + 8x_2 + 4x_3 + 14x_4 \geq 55 \quad \text{protein} \\ & 2x_1 + 285x_2 + 22x_3 + 80x_4 \geq 800 \quad \text{calcium} \\ & x_1, x_2, x_3, x_4 \geq 0 \end{aligned}\]

"""
对偶单纯形法
min z=15*x1+24*x2+5*x36*x2+x3>=25*x1+2*x2+x3>=1x1,x2>=0
标准化后格式
max z=-15*x1-24*x2-5*x3-6*x2-x3+x4=-2-5*x1-2*x2-x3+x5=-1x1,x2,x3,x4,x5>=0     
"""
import numpy as np# 根据线性规划要改变的数据
Cj = [-3, -9, -20, -19, 0, 0, 0]  # Coefficients of the objective function
constraints_matrix = [[-2000, -110, -160, -420, -260, 1, 0, 0],[-55, -4, -8, -4, -14, 0, 1, 0],[-800, -2, -285, -22, -80, 0, 0, 1]
]  # Constraints matrix (with RHS)
z = [0, -3, -9, -20, -19, 0, 0, 0]  # Z-row
Xb = [5, 6, 7]  # Basis variables (indices of slack variables)
Cb = [0, 0, 0]  # Coefficients of basis variables# 为便于表示而生成
C = [0] + Cj# Output simplex table
def show():print("--" * 50)print("|               Cj               ", end='|')for i in Cj:print(f"{i:.2f}".center(10), end='|')print()print("--" * 50)print("|    Cb    |    Xb    |    b     ", end='|')for i in range(len(Cj)):print(("X" + str(i + 1)).center(10), end='|')print()print("--" * 50)for i in range(len(constraints_matrix)):print("|", end="")print(f"{Cb[i]:.2f}".center(10), end='|')print(("X" + str(Xb[i])).center(10), end='|')for j in range(len(constraints_matrix[0])):print(f"{constraints_matrix[i][j]:.2f}".center(10), end='|')print()print("--" * 50)print("|          Z          ", end='|')for i in range(len(z)):print(f"{z[i]:.2f}".center(10), end='|')print()print("--" * 50)print("**" * 50)def fu():global constraints_matrix, z, Xb, Cb  # Declare global variables# Extracting the first columnfirst_column = [row[0] for row in constraints_matrix]# Finding the index of the minimum value in the first columnmin_index = first_column.index(min(first_column))# Extract the row corresponding to min_indexmin_row = constraints_matrix[min_index]# Calculate the ratios z / min_rowratios = [z[i] / min_row[i] if min_row[i] != 0 else float('inf') for i in range(len(z))]# Filter out the non-positive values from z and corresponding min_rowfiltered_ratios = [ratios[i] for i in range(len(ratios)) if z[i] < 0 and min_row[i] < 0]# Find the minimum ratio and its index in zif filtered_ratios:min_ratio = min(filtered_ratios)min_ratio_index = ratios.index(min_ratio)else:print("No valid ratio found")return# Updating XbXb[min_index] = min_ratio_index print("Updated basis variables Xb:")print(Xb)# Updating Cb based on XbCb = [C[Xb[i]] for i in range(len(Xb))]print("Updated coefficients of basis variables Cb:")print(Cb)# 根据 Xb 提取列以形成矩阵 BB = [[row[xb] for xb in Xb] for row in constraints_matrix]try:# Calculate the inverse of matrix BB_matrix = np.array(B)B_inv = np.linalg.inv(B_matrix)except np.linalg.LinAlgError:print("Matrix B is singular, skipping this iteration.")return# Update zz_update = (np.array(C[1:]) - np.dot(np.array(Cb), np.dot(B_inv, np.array(constraints_matrix)[:, 1:]))).tolist()z = [z[0]] + z_update  # Add the first element back to z# Update constraints_matrix with the inverse of Bconstraints_matrix = np.dot(B_inv, np.array(constraints_matrix)).tolist()# 输出最优解和最优值
def print_optimal_solution():# 最优解:基变量的值optimal_solution = [0] * len(Cj)  # 初始所有变量的值为0for i, xb in enumerate(Xb):optimal_solution[xb - 1] = constraints_matrix[i][0]  # 设置基变量的值# 最优值:计算目标函数值optimal_value = sum(Cj[i] * optimal_solution[i] for i in range(len(Cj)))optimal_value = -optimal_value  # 取相反数# 打印最优解和最优值,保留两位小数print(f"最优解(变量的取值):{[round(val, 2) for val in optimal_solution]}")print(f"最优值(目标函数值的相反数):{optimal_value:.2f}")# 迭代执行对偶单纯形法
while any(row[0] < 0 for row in constraints_matrix):fu()show()# 输出最后的最优解和最优值
print_optimal_solution()
#最终的单纯形表
Updated basis variables Xb:
[1, 6, 2]
Updated coefficients of basis variables Cb:
[-3, 0, -9]
----------------------------------------------------------------------------------------------------
|               Cj               |  -3.00   |  -9.00   |  -20.00  |  -19.00  |   0.00   |   0.00   |   0.00   |
----------------------------------------------------------------------------------------------------
|    Cb    |    Xb    |    b     |    X1    |    X2    |    X3    |    X4    |    X5    |    X6    |    X7    |
----------------------------------------------------------------------------------------------------
|  -3.00   |    X1    |  14.24   |   1.00   |   0.00   |   3.74   |   1.98   |  -0.01   |   0.00   |   0.01   |
|   0.00   |    X6    |  23.63   |   0.00   |   0.00   |  11.38   |  -3.96   |  -0.04   |   1.00   |  -0.01   |
|  -9.00   |    X2    |   2.71   |   0.00   |   1.00   |   0.05   |   0.27   |   0.00   |   0.00   |  -0.00   |
----------------------------------------------------------------------------------------------------
|          Z          |   0.00   |   0.00   |   0.00   |  -8.31   |  -10.67  |  -0.03   |   0.00   |  -0.02   |
----------------------------------------------------------------------------------------------------最优解(变量的取值):[14.24, 2.71, 0, 0, 0, 23.63, 0]
最优值(目标函数值的相反数):67.10

总结

单纯形法和对偶单纯形法是线性规划中最常用的两种算法,它们在解决优化问题方面发挥着关键作用。单纯形法通过从一个初始的基本可行解出发,沿着多面体的边逐步移动,寻找使目标函数值不断改善的邻近顶点,直到找到最优解。相较之下,对偶单纯形法则采用了不同的思路。它从一个对偶可行解开始,重点在于原问题的对偶问题的可行性,而非原问题的直接可行性。通过逐步调整,使得对偶条件和原始问题的可行性条件同时得到满足,从而找到最优解。对偶单纯形法特别适用于那些初始解不满足原问题约束但满足对偶约束的情况。这在需要快速适应变化的约束条件或在求解过程中频繁调整约束的场景中具有明显优势。
两种方法的有效结合和运用,可以大幅提升线性规划问题的求解效率和灵活性。理解和掌握这两种方法对于优化问题的解决至关重要,它不仅帮助优化计算速度和准确性,还能在面对不同类型的线性规划问题时,选择最适合的算法。无论是在理论研究还是实际应用中,这两种方法都提供了强大的工具,帮助我们应对各种复杂的决策问题。

参考文献

  1. 线性规划-单纯形算法详解
  2. 对偶单纯形法

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/790487.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ppt或wps安装islide

windows安装包下载: 官网:https://www.islide.cc/2. 一路下一步,可选择自定义安装路径,安装后点击体验,打开这个文件 3.能看到里面自动多了islide插件,内容可使用 以上仅供参考,如有疑问,留言联系

一键下载微博美图,‌这款浏览器插件让你轻松拥有!‌

偶尔在逛微博的时候,会遇到一些不错的照片,会想要保存下。但逐个保存太麻烦了,因此在 Chrome 插件商店搜了下,还真有一个能一键下载的插件,来分享一波。300.一键下载微博图片和视频 偶尔在逛微博的时候,会遇到一些不错的照片,会想要保存下。但逐个保存太麻烦了,因此在 …

使用 nuxi clean 命令清理 Nuxt 项目

title: 使用 nuxi clean 命令清理 Nuxt 项目 date: 2024/9/1 updated: 2024/9/1 author: cmdragon excerpt: nuxi clean 命令是管理和维护 Nuxt 项目的重要工具,它帮助你快速清理生成的文件和缓存,确保开发环境的干净。通过定期使用这个命令,你可以避免由于缓存或生成文件…

Dify大语言模型应用开发平台新手必备:安装注册与私有服务器部署全步骤

Dify简介 Dify是一个开源的大语言模型(Large Language Model, LLM)应用开发平台。它融合了后端即服务(Backend as a Service, BaaS)和LLMOps的理念,旨在帮助开发者,甚至是非技术人员,能够快速搭建和部署生成式AI应用程序。 Dify的主要特点包括:简化开发流程:通过提供一…

信息学奥赛初赛天天练-81-NOIP2015普及组-完善程序-二分答案、二分查找、中位数、二分边界、二分时间复杂度

1 完善程序 (单选题 ,每小题3分,共30分) 中位数 median 给定 n(n为奇数且小于 1000)个整数,整数的范围在 0∼m(0<m<2^31) 之间,请使用二分法求这 n个整数的中位数。所谓中位数,是指将这 n个数排序之后,排在正中间的数。(第五空 2 分,其余 3 分) 01 #include <…

Mac版Sourcetree暂存代码和取出代码

实际开发中经常遇到开发一半,要拉代码或者切分支的情况,这时候开发一半的代码如果不提交或者删除重置是无法拉取和切换分支的,那么这个时候可以把这部分代码暂存起来,然后在想取出的时候取出就行了 1.点击暂存文件,如下图2.点击贮藏,然后输入一个标识3.此时就可以正常拉取…

WPF Material Design中资源的查找和使用

Material Design中,一共分为两大块。一个是颜色资源,一个是控件资源。 下面来说下,如何使用控件资源: 在VS中,通过Nuget添加完Material Design 后,还需要在App.xaml中引用这些资源, 引用的方法如下图所示:在1处,引入material design的引用。 在2处,可以修改项目的主题…

对称二叉树-101

题目描述 给你一个二叉树的根节点 root , 检查它是否轴对称。解题思路 这里我们相当于是比较根节点左右两颗子树,我们依次向左右子树的左右两个方向进行遍历,我们比较左子树的左孩子和右子树的右孩子,左子树的右孩子和右子树的左孩子,这里如果不好理解可以看下面这个图片,…

数组更加深入的学习

1.浅了解java数组原理 可以形象的认为java中有“栈”和“堆”这两个东西,栈用于存放声明的数组,而堆则用于存放数组的赋值 刚声明完的数组没有任何作用,必须要对其赋值才有意义2.数组的三种初始化静态初始化:将数组的值在声明时提前写好并且值固定不变 动态初始化:声明并创…

vue学习---vue内置组件---keep-alive

什么是keep-alive?​ keep-alive是vue的一个 内置组件,能将不活动的组件实例保存在内存中,而不是直接将其销毁,它是一个抽象组件,不会被渲染到真实 DOM 中。keep-alive主要用于保存组件的渲染状态,避免组件反复创建和渲染,有效提升系统性能。keep-alive核心LRU算法 LRU(…

【闲话】OI计划

教练让的。顺手水一期闲话嘻嘻分数目标 1.1 初赛:S组60 1.2 复赛:S组150+ 1.3 NOIP:150+规划 2.1 9.1~9.10:复习图论,做完最短路题单 2.2 9.10~9.20:复习组合数学和代码阅读 2.3 9.22~10.25:学习数论、复习线段树等常用模板 2.4 10.27~11.29:总复习,背诵《骗分导论》(…

C# 定时器 Timer 如何精确到 1-2 毫秒以内

最近在排查项目OTA的一个问题,触发了一毫秒或者2毫秒执行一次进程间通信的,导致通信阻塞的问题。这样就需要用到模拟触发1ms或者2ms触发事件。这让我第一时间想到了C#的定时器。由于我们项目用到的框架是基于.NETFramwork4.8的,所以我就建立了一个.NETFramwork4.8的WPF Demo…