▶ 书中第六章部分程序,加上自己补充的代码,包括单纯形法求解线性规划问题

● 单纯形法求解线性规划问题

 // 表上作业法,I 为单位阵,y 为对偶变量,z 为目标函数值
// n m 1
// ┌───────────┬───────┬───┐
// │ │ │ │
// m │ A │ I │ b │
// a[m+1][n+m+1] = │ │ │ │
// ├───────────┼───────┼───┤
// 1 │ c │ y │ z │
// └───────────┴───────┴───┘ package package01; import edu.princeton.cs.algs4.StdOut;
import edu.princeton.cs.algs4.StdRandom; public class class01
{
private static final double EPSILON = 1.0E-10;
private double[][] a; // 工作表
private int m; // 约束数
private int n; // 变量数
private int[] basis; // basis[p] = q,第 p 行的基变量是 x[q] public class01(double[][] A, double[] b, double[] c)
{
m = b.length;
n = c.length;
for (int i = 0; i < m; i++)
{
if (b[i] < 0) // 要求 b 的分量均不小于 0
throw new IllegalArgumentException("RHS must be nonnegative");
}
a = new double[m + 1][n + m + 1];
for (int i = 0; i < m; i++) // a[0:m-1][0:n-1] = A(两边包含)
{
for (int j = 0; j < n; j++)
a[i][j] = A[i][j];
}
for (int i = 0; i < m; i++) // a[0:m-1][n:n+m-1] = I_m
a[i][n + i] = 1.0;
for (int i = 0; i < n; i++) // a[m][0:n-1] = c
a[m][i] = c[i];
for (int i = 0; i < m; i++) // a[0:m-1][n+m] = b
a[i][m + n] = b[i];
basis = new int[m];
for (int i = 0; i < m; i++)
basis[i] = n + i; for (;;) // 单纯形法求解
{
int q = -1;
for (int i = 0; q == -1 && i < m + n; i++) // 找到可优化的变量对应的列 q,即 c[j] > 0 的最靠前索引
q = (a[m][i] > 0) ? i : q;
if (q == -1) // 优化已完成
break; int p = -1; // 依最小比例规则选择离开向量 p
for (int i = 0; i < m; i++) // 要求 a[i][q] > 0 且要么 p 选第一个,要么选壁纸最大的那个
{
if (a[i][q] > EPSILON && (p == -1 || (a[i][m + n] / a[i][q]) < (a[p][m + n] / a[p][q])))
p = i;
}
if (p == -1) // 无离开向量,无界解
throw new ArithmeticException("Linear program is unbounded"); for (int i = 0; i <= m; i++) // 对其他行使用a[p][q] 进行高斯消元
{
for (int j = 0; j <= m + n; j++)
{
if (i != p && j != q)
a[i][j] -= a[p][j] * a[i][q] / a[p][q];
}
}
for (int i = 0; i <= m; i++) // 消元后其他行清零(消除浮点计算误差)
{
if (i != p)
a[i][q] = 0.0;
}
for (int j = 0; j <= m + n; j++)// 主元所在行归一化
{
if (j != q)
a[p][j] /= a[p][q];
}
a[p][q] = 1.0; basis[p] = q; // 更新基向量
}
assert check(A, b, c); // 检查结果
} private int dantzig() // 搜索 c 中值大于零的、最靠前的项的索引
{
int q = 0;
for (int j = 1; j < m + n; j++)
{
if (a[m][j] > a[m][q])
q = j;
}
return a[m][q] <= 0 ? -1 : q; // c 中各项均小于0,优化已完成
} public double value() // 最优解的目标函数值
{
return -a[m][m + n];
} public double[] primal() // 最优解的自变量值
{
double[] x = new double[n];
for (int i = 0; i < m; i++)
{
if (basis[i] < n)
x[basis[i]] = a[i][m + n];
}
return x;
} public double[] dual() // 最优解的对偶变量值
{
double[] y = new double[m];
for (int i = 0; i < m; i++)
y[i] = -a[m][n + i];
return y;
} private boolean isPrimalFeasible(double[][] A, double[] b) // 解的松弛性
{
double[] x = primal();
for (int j = 0; j < x.length; j++) // 检查最优解分量是否均非负
{
if (x[j] < 0.0)
{
StdOut.println("x[" + j + "] = " + x[j] + " is negative");
return false;
}
}
for (int i = 0; i < m; i++) // 检查约束条件
{
double sum = 0.0;
for (int j = 0; j < n; j++)
sum += A[i][j] * x[j];
if (sum > b[i] + EPSILON)
{
StdOut.println("not primal feasibleb\nb[" + i + "] = " + b[i] + ", sum = " + sum);
return false;
}
}
return true;
} private boolean isDualFeasible(double[][] A, double[] c) // 对偶解的松弛性
{
double[] y = dual();
for (int i = 0; i < y.length; i++) // 检查对偶变量不小于 0
{
if (y[i] < 0.0)
{
StdOut.println("y[" + i + "] = " + y[i] + " is negative");
return false;
}
}
for (int j = 0; j < n; j++) // 检查对偶约束条件 A*y ≥ c
{
double sum = 0.0;
for (int i = 0; i < m; i++)
sum += A[i][j] * y[i];
if (sum < c[j] - EPSILON)
{
StdOut.println("not dual feasible\nc[" + j + "] = " + c[j] + ", sum = " + sum);
return false;
}
}
return true;
} private boolean isOptimal(double[] b, double[] c) // 检查解的最优性 value == c*x == y*b
{
double[] x = primal(), y = dual();
double value = value(), value1 = 0.0;
for (int j = 0; j < x.length; j++)
value1 += c[j] * x[j];
double value2 = 0.0;
for (int i = 0; i < y.length; i++)
value2 += y[i] * b[i];
if (Math.abs(value - value1) > EPSILON || Math.abs(value - value2) > EPSILON)
{
StdOut.println("value = " + value + ", cx = " + value1 + ", yb = " + value2);
return false;
}
return true;
} private boolean check(double[][]A, double[] b, double[] c)
{
return isPrimalFeasible(A, b) && isDualFeasible(A, c) && isOptimal(b, c) && dantzig() == -1;
} private void show() // 输出工作表
{
StdOut.println("m = " + m);
StdOut.println("n = " + n);
for (int i = 0; i <= m; i++)
{
for (int j = 0; j <= m + n; j++)
StdOut.printf("%7.2f ", a[i][j]);
StdOut.println();
}
StdOut.println("value = " + value());
for (int i = 0; i < m; i++)
{
if (basis[i] < n)
StdOut.println("x_" + basis[i] + " = " + a[i][m + n]);
}
StdOut.println();
} private static void test(double[][] A, double[] b, double[] c) // 测试函数
{
class01 lp;
try // 有解正常输出,无解用异常提前退出
{
lp = new class01(A, b, c);
}
catch (ArithmeticException e)
{
System.out.println(e);
return;
}
StdOut.println("value = " + lp.value());
double[] x = lp.primal();
for (int i = 0; i < x.length; i++)
StdOut.println("x[" + i + "] = " + x[i]);
double[] y = lp.dual();
for (int j = 0; j < y.length; j++)
StdOut.println("y[" + j + "] = " + y[j]);
} private static void test1()
{
double[][] A = { { -1, 1, 0 },{ 1, 4, 0 },{ 2, 1, 0 },{ 3, -4, 0 },{ 0, 0, 1 } };
double[] b = { 5, 45, 27, 24, 4 }, c = { 1, 1, 1 };
test(A, b, c);
} private static void test2() // x[0] = 12, x[1] = 28, value = 800
{
double[][] A = { { 5.0, 15.0 },{ 4.0, 4.0 },{ 35.0, 20.0 } };
double[] b = { 480.0, 160.0, 1190.0 }, c = { 13.0, 23.0 };
test(A, b, c);
} private static void test3() // unbounded
{
double[][] A = { { -2.0, -9.0, 1.0, 9.0 },{ 1.0, 1.0, -1.0, -2.0 } };
double[] b = { 3.0, 2.0 }, c = { 2.0, 3.0, -1.0, -12.0 };
test(A, b, c);
} private static void test4() // x[0] = 1.0, x[1] = 0.0, x[2] = 1.0, x[3] = 0.0, value = 1.0,周期
{
double[][] A = { { 0.5, -5.5, -2.5, 9.0 },{ 0.5, -1.5, -0.5, 1.0 },{ 1.0, 0.0, 0.0, 0.0 } };
double[] b = { 0.0, 0.0, 1.0 }, c = { 10.0, -57.0, -9.0, -24.0 };
test(A, b, c);
} public static void main(String[] args)
{
StdOut.println("----- test 1 --------------------");
test1();
StdOut.println("\n----- test 2 --------------------");
test2();
StdOut.println("\n----- test 3 --------------------");
test3();
StdOut.println("\n----- test 4 --------------------");
test4();
StdOut.println("\n----- test random ---------------"); // 随机测试
int m = Integer.parseInt(args[0]), n = Integer.parseInt(args[1]);
double[][] A = new double[m][n];
double[] b = new double[m], c = new double[n];
for (int i = 0; i < m; i++)
{
for (int j = 0; j < n; j++)
A[i][j] = StdRandom.uniform(100);
}
for (int i = 0; i < m; i++)
b[i] = StdRandom.uniform(1000);
for (int j = 0; j < n; j++)
c[j] = StdRandom.uniform(1000);
class01 lp = new class01(A, b, c);
test(A, b, c);
}
}

《算法》第六章部分程序 part 8的更多相关文章

  1. 《算法》第六章部分程序 part 7

    ▶ 书中第六章部分程序,加上自己补充的代码,包括全局最小切分 Stoer-Wagner 算法,最小权值二分图匹配 ● 全局最小切分 Stoer-Wagner 算法 package package01; ...

  2. 《算法》第六章部分程序 part 6

    ▶ 书中第六章部分程序,包括在加上自己补充的代码,包括二分图最大匹配(最小顶点覆盖)的交替路径算法和 HopcroftKarp 算法 ● 二分图最大匹配(最小顶点覆盖)的交替路径算法 package ...

  3. 《算法》第六章部分程序 part 5

    ▶ 书中第六章部分程序,包括在加上自己补充的代码,网络最大流 Ford - Fulkerson 算法,以及用到的流量边类和剩余流量网络类 ● 网络最大流 Ford - Fulkerson 算法 pac ...

  4. 《算法》第六章部分程序 part 4

    ▶ 书中第六章部分程序,包括在加上自己补充的代码,利用后缀树查找最长重复子串.查找最大重复子串并输出其上下文(Key word in context,KWIC).求两字符串的最长公共子串 ● 利用后缀 ...

  5. 《算法》第六章部分程序 part 3

    ▶ 书中第六章部分程序,包括在加上自己补充的代码,后缀树的两种实现 ● 后缀树实现一 package package01; import java.util.Arrays; import edu.pr ...

  6. 《算法》第六章部分程序 part 2

    ▶ 书中第六章部分程序,包括在加上自己补充的代码,B-树 ● B-树 package package01; import edu.princeton.cs.algs4.StdOut; public c ...

  7. 《算法》第六章部分程序 part 1

    ▶ 书中第六章部分程序,包括在加上自己补充的代码,粒子碰撞系统及用到的粒子类 ● 粒子系统 package package01; import java.awt.Color; import edu.p ...

  8. 《算法》第一章部分程序 part 1

    ▶ 书中第一章部分程序,加上自己补充的代码,包括若干种二分搜索,寻找图上连通分量数的两种算法 ● 代码,二分搜索 package package01; import java.util.Arrays; ...

  9. 《算法》第二章部分程序 part 5

    ▶ 书中第二章部分程序,加上自己补充的代码,包括利用优先队列进行多路归并和堆排序 ● 利用优先队列进行多路归并 package package01; import edu.princeton.cs.a ...

随机推荐

  1. Emacs的一些事情(与Vi的争议及使用)

    一年成为Emacs高手(像神一样使用编辑器)推荐文章 http://ftp.gnu.org/gnu/emacs/windows/http://blog.csdn.net/redguardtoo/art ...

  2. 解析H5本地储存Web Storage

    一.本地存储由来的背景 由于HTML4时代Cookie的大小.格式.存储数据格式等限制,网站应用如果想在浏览器端存储用户的部分信息,那么只能借助于Cookie.但是Cookie的这些限制,也就导致了C ...

  3. 服务容错保护断路器Hystrix之六:服务熔断和服务降级

    伴随着微服务架构被宣传得如火如荼,一些概念也被推到了我们面前(管你接受不接受),其实大多数概念以前就有,但很少被提的这么频繁(现在好像不提及都不好意思交流了).想起有人总结的一句话,微服务架构的特点就 ...

  4. Scala函数式对象-有理数

    有理数类的表示 实现规范:支持有理数的加减乘除,并支持有理数的规范表示 1.定义Rational 首先,考虑用户如何使用这个类,我们已经决定使用“Immutable”方式来使用Rational对象,我 ...

  5. 廖雪峰Java5集合-1Java集合简介-1Java结合简介

    1.集合 定义:集合就是一堆东西.集合里的东西,称为元素Element 数学中的集合: 有限集合: * 一个班所有的学生组成的集合 * 一个网站所有的商品组成的集合 无限集合: * 全体自然数集合 * ...

  6. centos7 搭建DHCP服务器

    一.DHCP简单讲解 DHCP就是动态主机配置协议(Dynamic Host Configuration Protocol)是一种基于UDP协议且仅限用于局域网的网络协议,它的目的就是为了减轻TCP/ ...

  7. fragment--的生命周期

    官网帮助文档链接: http://developer.Android.com/guide/components/fragments.html 主要看两张图,和跑代码 1,Fragment的生命周: 2 ...

  8. is not valid JSON: json: cannot unmarshal string into Go value of type map[string]interface | mongodb在windows和Linux导出出错

    执行mongoexport命令的时候 mongoexport --csv -f externalSeqNum,paymentId --host 127.0.0.1:27017 -d liveX -c ...

  9. [UE4]声音系统概述

    一.只能使用wav格式的声音 二.wav声音可以直接播放到打开的UE4编辑器内打开的Content文件夹.也可以直接导入 三.在Content中的文件夹的声音资源可以直接拖放到场景中,会以3D场景声音 ...

  10. javascript将list转换成树状结构

    /** * 将list装换成tree * @param {Object} myId 数据主键id * @param {Object} pId 数据关联的父级id * @param {Object} l ...