算法导论一书的第四部分—高级设计和分析技术从本章开始讨论,主要分析高效算法的三种重要技术:动态规划、贪心算法以及平摊分析三种。

首先,本章讨论动态规划,它是通过组合子问题的解而解决整个问题的,通常应用于最优化问题。

动态规划算法的设计可以分为如下4个步骤:

  • 描述最优解的结构
  • 递归定义最优解的值
  • 按照自底向上的方式计算最优解的值
  • 由计算出的结果构造一个最优解

15.1 装配线调度

问题描述

第一个动态规划的例子是求解一个制造问题,Colonel汽车公司在有两条装配线的工厂生产汽车,具体如下图所示:



针对装配线调度问题,下面给出一个实例,要求利用动态规划方法分析设计程序实现。

程序实现

/*
* 《算法导论》第十五章 动态规划
* 15.1 装配线调度问题
*/ #include <iostream>
#include <cstdlib> using namespace std; //初始化装配线以及站点个数
const int L = 2;
const int S = 6; //程序输入:装配点耗费时间、底盘进入耗费时间、装配点移动耗费时间、底盘离开耗费时间
int a[L + 1][S + 1] , e[L + 1], t[L + 1][S], x[L + 1] , n = S; //程序输出:
int f[L + 1][S + 1], l[L + 1][S + 1] , rf , rl; //求解f[][] 用到的参数int **a, int *e, int **t, int *x, int n
void FastestWay()
{
f[1][1] = e[1] + a[1][1];
f[2][1] = e[2] + a[2][1];
//由公式15.4 - 15.7 推导计算 for (int j = 2; j <= n; j++)
{
if (f[1][j-1] + a[1][j] <= f[2][j - 1] + t[2][j - 1] + a[1][j])
{
f[1][j] = f[1][j - 1] + a[1][j];
l[1][j] = 1;
}
else{
f[1][j] = f[2][j - 1] + t[2][j - 1] + a[1][j];
l[1][j] = 2;
} if (f[2][j - 1] + a[2][j] <= f[1][j - 1] + t[1][j - 1] + a[2][j])
{
f[2][j] = f[2][j - 1] + a[2][j];
l[2][j] = 2;
}
else{
f[2][j] = f[1][j - 1] + t[1][j - 1] + a[2][j];
l[2][j] = 1;
}
} if (f[1][n] + x[1] <= f[2][n] + x[2])
{
rf = f[1][n] + x[1];
rl = 1;
}
else{
rf = f[2][n] + x[2];
rl = 2;
} } //打印调度结果
void PrintStations()
{
//最后一个站点的装配线号
int i = rl; cout << "倒序结果输出:" << endl; cout << "line : " << i << " , station : " << n << endl; for (int j = n; j >= 2; j--)
{
i = l[i][j];
cout << "line : " << i << " , station : " << j - 1 << endl;
}
} //打印调度结果 —— 以站号递增的顺序输出装配站
void PrintStations2()
{
int r[S] , tem = rl; for (int j = S; j > 1 ; j--)
{
tem= l[tem][j];
r[j - 1] = tem;
} cout << "正序结果输出:" << endl; for (int j = 1; j < S; j++)
{
cout << "line : " << r[j] << " , station : " << j << endl;
}
//输出最后一个装配站
cout << "line : " << rl << " , station : " << n << endl;
} void Input()
{
cout << "请输入装配点耗费时间:" << endl;
for (int i = 1; i <= L; i++)
for (int j = 1; j <= S; j++)
{
cin >> a[i][j];
} cout << "请输入进入装配点耗费时间:" << endl;
for (int i = 1; i <= L; i++)
{
cin >> e[i];
} cout << "请输入站点移动耗费时间:" << endl;
for (int i = 1; i <= L; i++)
for (int j = 1; j < S; j++)
{
cin >> t[i][j];
}
cout << "请输入离开装配点耗费时间:" << endl;
for (int i = 1; i <= L; i++)
{
cin >> x[i];
}
} void Output()
{
int i, j;
cout << "输出f[i][j]" << endl;
//f[i][j]表示第j个站是在装配线i上完成的,完成1到j的装配最少需要的时间
for (i = 1; i <= L; i++)
{
for (j = 1; j <= S; j++)
cout << f[i][j] << ' ';
cout << endl;
} cout << "rf = " << rf << endl; cout << "输出l[i][j]" << endl;
//l[i][j]表示使得f[i][j]最小时在哪个装配线上装配j-1
for (i = 1; i <= L; i++)
{
for (j = 2; j <= S; j++)
cout << l[i][j] << ' ';
cout << endl;
} cout << "rl = " << rl << endl;
} //主程序
int main()
{
//数据输入
Input(); FastestWay(); //数据输出
Output(); //结果打印
PrintStations(); PrintStations2(); system("pause");
return 0;
}

运行结果

15.1练习

15.1-1

//递归输出结果,参数为:(站点,装配线编号)
void PrintStations3(int s, int ll)
{
cout << "正序结果输出:" << endl; if (s < n)
ll = l[ll][s + 1]; if (s > 1)
PrintStations3(s - 1, ll); cout << "line : " << ll << " , station : " << s << endl;
}

15.1-5

若Canty教授猜测正确,存在满足l1[j]=2且l2[j]=1的站点,则公式15.6与公式15.7中的公式必须同时成立,得到:

f1[j−1]+a1,j>f2[j−1]+t2,j−1+a1,j
f2[j−1]+a2,j>f1[j−1]+t1,j−1+a2,j

明显的,为两个矛盾式。所以Canty教授的猜测是不正确的。

15.2 矩阵链乘法

问题描述

解决矩阵链乘问题是动态规划的一个典型实例。

程序实现

/*
* 《算法导论》第十五章 动态规划
* 矩阵链乘法
*/ #include <iostream>
#include <cstdlib> using namespace std; const int N = 6; //进行链乘的矩阵数目 //链乘矩阵维数序列
int A[N + 1] = { 30, 35, 15, 5, 10, 20, 25 };
//int A[N + 1] = { 5, 10, 3, 12, 5, 50, 6 };
//动态优化结果存储
int m[N + 1][N + 1] , s[N+1][N+1]; void MatrixChainOrder()
{
int i, j, k , l, q;
for (i = 1; i <= N; i++)
m[i][i] = 0; for (l = 2; l <= N; l++)
{
for (i = 1; i <= N - l + 1; i++)
{
j = i + l - 1;
m[i][j] = 0x7fffffff;
for (k = i; k <= j - 1; k++)
{
q = m[i][k] + m[k + 1][j] + A[i - 1] * A[k] * A[j];
if (q < m[i][j])
{
m[i][j] = q;
s[i][j] = k;
}
}
}//for
}
} void PrintOptimalParens(int i , int j)
{
if (i == j)
cout << "A" << i << " ";
else{
cout << "(";
PrintOptimalParens( i, s[i][j]);
PrintOptimalParens( s[i][j] + 1, j);
cout << ")";
}
} int main()
{
MatrixChainOrder();
PrintOptimalParens(1, N);
cout << endl;
system("pause");
return 0;
}

运行结果

15.2练习

15.2-1

最优加括号结果为:

15.2-2

15.3 动态规划基础

采用动态规划方法的最优化问题中的两个要素:最优子结构和重叠子问题。

  1. 最优子结构

    用动态规划求解优化问题的第一步是描述最优解的结构,如果问题的一个最优解中包含了子问题的最优解,则该问题具有最优子结构。

2.重叠子问题

适用于动态规划求解的最优化问题必须具有的第二个要素是子问题的空间要“很小”,也就是用来解原问题的递归算法可以反复的解同样的子问题,而不是总在产生新的子问题。

重叠子问题实现矩阵链乘

/*
* 《算法导论》第十五章 动态规划
* 15.3 动态规划基础
* 重叠子问题实现
*/ #include <iostream>
#include <cstdlib> using namespace std; const int N = 6; //进行链乘的矩阵数目 //动态优化结果存储
int m[N + 1][N + 1], s[N + 1][N + 1]; int RecusiveMatrixChain(int *A, int i, int j)
{
if (i == j)
return 0;
m[i][j] = 0x7fffffff; for (int k = i; k <= j - 1; k++)
{
int q = RecusiveMatrixChain(A, i, k) + RecusiveMatrixChain(A, k + 1, j) + A[i - 1] * A[k] * A[j]; if (q < m[i][j])
{
m[i][j] = q;
s[i][j] = k;
} }//for
return m[i][j];
} void PrintOptimalParens(int i, int j)
{
if (i == j)
cout << "A" << i << " ";
else{
cout << "(";
PrintOptimalParens(i, s[i][j]);
PrintOptimalParens(s[i][j] + 1, j);
cout << ")";
}
} int main()
{
//链乘矩阵维数序列
int A[N + 1] = { 30, 35, 15, 5, 10, 20, 25 };
//int A[N + 1] = { 5, 10, 3, 12, 5, 50, 6 }; RecusiveMatrixChain(A, 1, N);
PrintOptimalParens(1, N); cout << endl;
system("pause");
return 0;
}

做备忘录实现矩阵链乘

动态规划有一种变形,它既具有通常的动态规划方法的效率,又采用了一种自顶向下的策略。其思想就是备忘原问题的自然但低效的递归算法。

/*
* 《算法导论》第十五章 动态规划
* 15.3 动态规划基础
* 做备忘录
*/ #include <iostream>
#include <cstdlib> using namespace std; const int N = 6; //进行链乘的矩阵数目 //动态优化结果存储
int m[N + 1][N + 1], s[N + 1][N + 1]; int LookUpChain(int *A, int i, int j)
{ if (m[i][j] < 0x7fffffff)
return m[i][j]; if (i == j)
m[i][j] = 0;
else
{
for (int k = i; k <= j - 1; k++)
{
int q = LookUpChain(A, i, k) + LookUpChain(A, k + 1, j) + A[i - 1] * A[k] * A[j]; if (q < m[i][j])
{
m[i][j] = q;
s[i][j] = k;
} }//for
} return m[i][j];
} int MemorizedMatrixChain(int *A)
{
for (int i = 1; i <= N; i++)
for (int j = 1; j <= N; j++)
{
m[i][j] = 0x7fffffff;
} return LookUpChain(A, 1, N);
} void PrintOptimalParens(int i, int j)
{
if (i == j)
cout << "A" << i << " ";
else{
cout << "(";
PrintOptimalParens(i, s[i][j]);
PrintOptimalParens(s[i][j] + 1, j);
cout << ")";
}
} int main()
{
//链乘矩阵维数序列
int A[N + 1] = { 30, 35, 15, 5, 10, 20, 25 };
//int A[N + 1] = { 5, 10, 3, 12, 5, 50, 6 }; MemorizedMatrixChain(A);
PrintOptimalParens(1, N); cout << endl;
system("pause");
return 0;
}

15.3练习

《算法导论》— Chapter 15 动态规划的更多相关文章

  1. 算法导论——lec 11 动态规划及应用

    和分治法一样,动态规划也是通过组合子问题的解而解决整个问题的.分治法是指将问题划分为一个一个独立的子问题,递归地求解各个子问题然后合并子问题的解而得到原问题的解.与此不同,动态规划适用于子问题不是相互 ...

  2. 算法导论课后习题解答 第一部分 练习1.1-1->1.1-5

    很高兴能和大家一起共同学习算法导论这本书.笔者将在业余时间把算法导论后面的题解以博文的形式展现出来希望能得到大家的支持谢谢.如果有可能我会做一些教学视频免费的供大家观看. 练习题选自算法导论中文第三版 ...

  3. (搬运)《算法导论》习题解答 Chapter 22.1-1(入度和出度)

    (搬运)<算法导论>习题解答 Chapter 22.1-1(入度和出度) 思路:遍历邻接列表即可; 伪代码: for u 属于 Vertex for v属于 Adj[u] outdegre ...

  4. 《算法导论》 — Chapter 7 高速排序

    序 高速排序(QuickSort)也是一种排序算法,对包括n个数组的输入数组.最坏情况执行时间为O(n^2). 尽管这个最坏情况执行时间比較差.可是高速排序一般是用于排序的最佳有用选择.这是由于其平均 ...

  5. Demo003 最大连续子数组问题(《算法导论》4.1-5)

    问题 问题描述  给定n个整数(可能为负数)组成的数组,要求一个数组连续下标和的最大值,数组的元素可正.可负.可为零.  数组中连续的一个或多个整数组成一个子数组,每个子数组都有一个和.求所有子数组的 ...

  6. 《算法导论》 — Chapter 7 快速排序

    序 快速排序(QuickSort)也是一种排序算法,对包含n个数组的输入数组,最坏情况运行时间为O(n^2).虽然这个最坏情况运行时间比较差,但是快速排序通常是用于排序的最佳实用选择,这是因为其平均性 ...

  7. 红黑树——算法导论(15)

    1. 什么是红黑树 (1) 简介     上一篇我们介绍了基本动态集合操作时间复杂度均为O(h)的二叉搜索树.但遗憾的是,只有当二叉搜索树高度较低时,这些集合操作才会较快:即当树的高度较高(甚至一种极 ...

  8. 《算法导论》— Chapter 9 中位数和顺序统计学

    序 在算法导论的第二部分主要探讨了排序和顺序统计学,第六章~第八章讨论了堆排序.快速排序以及三种线性排序算法.该部分的最后一个章节,将讨论顺序统计方面的知识. 在一个由n个元素组成的集合中,第i个顺序 ...

  9. 《算法导论》 — Chapter 8 线性时间排序

    序 到目前为止,关于排序的问题,前面已经介绍了很多,从插入排序.合并排序.堆排序以及快速排序,每一种都有其适用的情况,在时间和空间复杂度上各有优势.它们都有一个相同的特点,以上所有排序的结果序列,各个 ...

随机推荐

  1. python之文件的读写

    读  r   读写模式   r+     如果打开文件时没有指定模式,默认只读,如果使用r或r+,文件不存在时会报错 写  w  写读模式  w+    w模式会清空原有的文件内容 追加 a  追加读 ...

  2. Hdu 4778 Gems Fight! (状态压缩 + DP)

    题目链接: Hdu 4778 Gems Fight! 题目描述: 就是有G种颜色,B个背包,每个背包有n个宝石,颜色分别为c1,c2............两个人轮流取背包放到公共容器里面,容器里面有 ...

  3. Educational Codeforces Round 46 (Rated for Div. 2) B. Light It Up

    Bryce1010模板 http://codeforces.com/problemset/problem/1000/B 思路:先用两个数组sumon[]和sumoff[]将亮着的灯和灭的灯累计一下. ...

  4. Codeforces Round #324 (Div. 2)

    CF的rating设置改了..人太多了,决定开小号打,果然是明智的选择! 水 A - Olesya and Rodion #include <bits/stdc++.h> using na ...

  5. Finally语句

  6. Learn More Study Less `my notes`

    整体性学习概念: 广泛扎实的基础知识 抽象知识成生活中的模型,便于记忆 融会贯通,创造新的东西 整体性学习组成 获取:积极阅读:标记并结合其他的知识点 主要观点 怎么记住:联系和比喻其他的知识 拓展和 ...

  7. WIN2003 IIS相关错误解决方案

    我碰到的主要问题是:“Server Application Unavailable 错误”.“无法显示网页”: 1.如果你的.NET版本是2.0及以上的话,那要注意了:win2003是默认安装1.1的 ...

  8. 责任链模式和php实现

    职责链模式(又叫责任链模式): 包含了一些命令对象和一些处理对象,每个处理对象决定它能处理那些命令对象,它也知道应该把自己不能处理的命令对象交下一个处理对象,该模式还描述了往该链添加新的处理对象的方法 ...

  9. 公有云大脑——核心IDC简影

    出差刚到家,公司最近接了一个矿场转建公有云平台的项目. 前期200台服务器作为公有云基础. 我主要负责总体网络规划.计费数据库集群设计.ceph集群自动部署.容器化设计.硬件及系统调试优化等等! 由于 ...

  10. fedora kde桌面系统配置

    本文向大家分享个人将fedora操作系统作为工作生活首选桌面系统的一些配置经验,系统版本与fedora最新版本保持一致,当前为fedora 25. #添加rpm源su -c 'dnf install ...