基本思想:把求解的问题分成许多阶段或多个子问题,然后按顺序求解各个子问题。**前一个子问题的解为后一个子问题的求解提供了有用的信息。**在求解任何一子问题时,列出各种可能的局部解,通过决策保留那些有可能达到最优的局部解,丢弃其它局部解,依次解决各子问题,最后一个子问题就是问题的解。
对于一个多阶段过程问题,最优策略是否存在,不仅依赖于该问题是否有最优子结构性质,而能否采取动态规划的方法,还要看该问题的子问题是否具有重叠性质。
最优子结构性质:原问题的最优解包含了其子问题的最优解。
子问题重叠性质:每次产生的子问题并不总是新问题,有些子问题被反复计算多次。
矩阵连乘问题
for(int *p,int n,int **t,int **s){
for(int i = 1; i <= n; i++) m[i][i] = 0;
for(int r = 2; r<= n; r++){
for(int i=1; i <= n-r+1; i++){
int j = i+r-1;
m[i][j] = m[i][i] + m[i+1][j] + p[i-1]*p[i]*p[j];
s[i][j] = i;
for(int k=i+1; k < j; k++){
int t = m[i][k] + m[k+1][j] + p[i-1]*p[k]*p[j];
if(t <= m[i][j]){
m[i][j] = t;
s[i][j] = k;
}
}
}
}
}
算法的计算时间上界O(n3),所占用的空间为O(n2)
由最长公共子序列问题的最优子结构性质建立子问题最优值的递归关系,用c[i][j]记录最长公共子序列的长度。
当i=0或j=0时,空序列是A和B的最长公共子序列。因此C[i][j]=0。
由于在所考虑的子问题空间中,总有θ(mn)个不同的子问题,因此用动态规划自底向上计算最优值能提高算法的效率。
void LCSLength(int m,int n,char *x,char *y,int **c,int **b){
int i,j;
for(i=1; i<=m; i++) c[i][0]=0;
for(i=1; i<=n; i++) c[0][i]=0;
for(i=1; i<=m; i++){
for(j=1; j<=n; j++){
if(x[i]==y[j]){
c[i][j] = x[i-1][j-1] + 1;
b[i][j] = 1;
}else if(c[i-1][j] >= c[i][j-1]){
c[i][j] = c[i-1][j];
b[i][j] = 2;
}else{
c[i][j] = c[i][j-1];
b[i][j] = 3;
}
}
}
}
构造最长公共子序列
void LCS(int i,int j,char *x,int **b){
if(i==0 || j==0) return;
if(b[i][j] == 1){
LCS(i-1,j-1,x,b);
printf("%d ",x[i]);
}else if(b[i][j] == 2){
LCS(i-1,j,x,b);
}else{
LCS(i,j-1,x,b);
}
}
根据序列x,y,建立两个(m+1)x(n+1)的二维表C和二维表B,分别存放搜索过程中得到的子序列的长度和状态。
令V(i,j)表示前i个物品,能够装入容量为j的背包中的物品最大值。
V(i,0) = 0; 背包容量为0
V(0,j) = 0; 没有物品
void KnapSack(int n,int w[],int v[][]){
int i,j;
for(i=0; i<=n; i++) v[i][0] = 0;
for(j=0; j<=c; j++) v[0][j] = 0;
for(i=1; i<=n; i++){
for(j=1; j<=c; j++){
if(w[i] > j){
v[i][j] = v[i-1][j];
}else{
v[i][j] = max{v[i-1][j] , v[i-1][j-w[i]]+v[i]};
}
}
}
// 求装入背包的物品
j = c;
for(i=n;i>=0;i--){
if(v[i][j] > v[i-1][j]){
x[i] = 1;
j = j - w[i];
}else{
x[i] = 0;
}
}
return v[n][c];
}
贪心算法总是做出在当前看来最好的选择,即所作的选择是局部最优解。
希望从局部的最优解得到整体最优解。
采用逐步构造最优解的方法。在每个阶段,都作出一个看上去最优的策略(在一定的标准下)。决策一旦做出,就不可以再更改。
贪心算法求解问题的基本要素
已知n个活动E={1,2,…,n}要求使用同一资源,第k个活动要求的开始时间和结束时间为sk和fk,其中sk
活动安排问题就是在所给的活动集合中选出最大的相容活动子集。
基本思路
在安排时将结束时间早的活动尽量往前安排,好给后面的活动安排留出更多的空间,从而达到安排最多活动的目标。
贪心准则:在未安排的活动中挑选结束时间最早的活动安排
首先将安排的11个活动的开始时间和结束时间按结束时间的非减次序列排序:
各活动的开始时间和结束时间已经按结束时间的非减序排列了。
public static void greedySelector(int s[], int f[], bool a[]){
int n = s.length - 1;
a[0] = true;
int j = 0; //已经安排的最后一个活动
int count = 1;
for(int i=1; i<=n; i++){
if(s[i] >= f[j]){
a[i] = true;
j = i;
count++;
}else{
a[i] = false;
}
}
return count;
}
0-1背包问题不能用贪心算法求解,因为0-1背包问题无法保证最终能将背包装满,部分闲置的背包空间使单位背包空间的价值降低了。
在选择物品i装入背包时,可以选择物品i的一部分,而不一定要全部装入背包。
用贪心算法解决背包问题的基本思想:
void Knapsack(int n, float M,float v[], float w[], float x[]){
sort(n,v,w);//将物品按单位重量价值排序
int i;
for(i=1; i<=n; i++) x[i] = 0;
float c = M;
for(i=1; i<=n; i++){
if(w[i] > c){
break;
}
x[i] = 1;
c = c-w[i];
}
if(i<=n){
x[i] = c/w[i];
}
}
该算法的主要计算时间在于将各种物品依此按单位重量价值从大到小排序,因此算法的计算时间上界为O(nlogn)