动态规划初步 刘汝佳字数 数字三角形

有一个由非负整数组成的三角形,第一行只有一个数,除了最下行之外每个数左下方和右下方各有一个数 如图所示

 

从第一行的数开始,每次可以往左下或右下走一格,直到走到最下行,把沿途经过的数全部加起来,如何走才能使得这个和最大?

分析:

一看到题目我们很自然的可以想到用回溯法(DFS)做,即每次都选择靠左的格子走,走到头后再回溯选择靠右的格子,按<左,右>的顺序遍历完所有的路径

代码如下:

#include<iostream>
using namespace std;

int a[11] = {0,29,9,5,12,7,26,6,14,15,8};//假设有四层
int max = -1, sum = 0;
int shuzi(int i, int j)//计算第i层第j个的数字在a数组的第几个
{
    int x = 0;
    for (int l = 1; l <= i - 1; l++)
    {
        x += l;
    }
    x += j;
    return x;
}
int dfs(int i, int j)
{
    if (i==5)
    {
        if (sum > max)
            max = sum;
        return 0;
    }
    sum += a[shuzi(i, j)];
    dfs(i + 1, j);
    dfs(i + 1, j + 1);
    sum -= a[shuzi(i, j)];
    return 0;
}
int main(int argc, char* argv[])
{
    dfs(1, 1);
    cout << max<<endl;
    return 0;
}

时间复杂度是2^n,指数级的,n一旦大了就会超时

所以用状态转移方程来发现每项的依赖项

方程如下:

d(i,j)=a(i,j)+max{d(i+1,j),d(i+1,j+1)};

代码如下:

#include<iostream>
using namespace std;

int a[20] = {0,1,2,3,4,5,6,7,8,9,12,0,0,0,0,0,0,0,0,0};
int max(int a, int b)
{
    return a > b ? a : b;
}
int shuzhi(int i, int j)
{
    int x = 0;
    for (int l = 1; l <= i - 1; l++)
    {
        x += l;
    }
    x += j;
    return x;
}
int solve(int i, int j)
{
    int x=shuzhi(i, j);
    return a[x]+(i==4?0:max(solve(i+1,j),solve(i+1,j+1)));
    
}
int main(int argc, char* argv[])
{
    cout << solve(1, 1)<<endl;
    return 0;
}

已经知道了每项的依赖项,就可以避免重复计算

#include<iostream>
using namespace std;

int a[20] = {0,13,11,8,12,7,26,6,14,15,8};
int dp[10][10] = { {0,0} };
int max(int a, int b)
{
    return a > b ? a : b;
}
int shuzi(int i, int j)
{
    int x = 0;
    for (int l = 1; l <= i - 1; l++)
    {
        x += l;
    }
    x += j;
    return x;
}
int solve(int i, int j)
{
    for (int k = 1; k <= 4; k++)
        dp[4][k] = a[shuzi(4, k)];
    for (int k = 3; k >= 1; k--)
        for (int l = 1; l <= k; l++)
            dp[k][l] = a[shuzi(k, l)]+ max(dp[k + 1][l], dp[k + 1][l + 1]);
    return dp[i][j];
}
int main(int argc, char* argv[])
{
    cout << solve(1, 1) << endl;
    return 0;
}

简化版:

#include<iostream>
using namespace std;

int a[20] = { 0,1,2,3,4,5,6,7,8,9,10};
int dp[10][10] = { { 0,0 } };
int max(int a, int b)
{
    return a > b ? a : b;
}
int shuzi(int i, int j)
{
    int x = 0;
    for (int l = 1; l <= i - 1; l++)
    {
        x += l;
    }
    x += j;
    return x;
}
int solve(int i, int j)
{
    if (dp[i][j] >= 0) return dp[i][j];
    return a[shuzi(i, j)] + (i==4?0:max(solve(i+1,j), solve(i + 1, j+1)));

}
int main(int argc, char* argv[])
{
    memset(dp, -1, sizeof(dp));
    cout << solve(1, 1) << endl;
    return 0;
}

 

posted @ 2018-11-07 18:39  uzi2008  阅读(208)  评论(0编辑  收藏  举报