个人对简单的dp问题的理解:找是否有重叠问题,明确递推关系,怎么推的(顺序千万不要搞错),找到状态方程,循环时注意边界条件和方程式是否严格成立。
转自:https://www.cnblogs.com/zyx1301691180/p/5727918.html
HDU 2084
有如下所示的数塔,要求从顶层走到底层,若每一步只能走到相邻的结点,则经过的结点的数字之和最大是多少?

已经告诉你了,这是个DP的题目,你能AC吗?
- 刻画一个最优解的结构特征。
- 递归地定义最优解的值。
- 计算最优解的值,通常采用自底向上的方法。
- 利用计算出的信息构造一个最优解。”
我觉得这道题的解题步骤就完全可以用这个来描述:
1.首先,一个最优解的结构特征,即从顶层走到底层,其各个节点的最大数字之和的为最优解。
2.题目中要求走法为自顶向下,且每一步只能走到相邻的节点,那么每个节点只有两种选择,即这个节点的两个子节点;
那么这个节点的最优解就等于这个节点的值加上其两个节点的最优解的最大值;
3.自底向上的求解。
4.我们所求出来的解就是我们所需要的答案,则第四步在这里可以忽略掉。
最后附上我的代码:
我用数组a来接收数塔的值,则第三部可以表示为:a[i][j]的最优解=max(a[i+1][j]的最优解,a[i+1][j+1]的最优解);
而数塔最底层节点的最优解就等于他本身的值,因为他只有一个;那么最底层的值已知,我们只需要从下往上递推就可以了
即这三行代码块:
for(i=n-1;i>=0;i--)
for(j=0;j<=i;j++)
a[i][j]=a[i][j]+max(a[i+1][j],a[i+1][j+1]);
#include <bits/stdc++.h>
using namespace std ;
int main ()
{
long long dp[][];
int T; cin>>T;
while (T--)
{
int n; cin>>n;
memset(dp,,sizeof(dp));
for(int i=; i<n; i++)
for(int j=; j<=i; j++)
cin>>dp[i][j]; for(int i=n-; i>=; i--)
for(int j=; j<=i; j++)
dp[i][j]+=max(dp[i+][j],dp[i+][j+]); cout<<dp[][]<<endl ;
}
return ;
}
//用滚动数组优化一下
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <string>
using namespace std; int n;
const int INF=0x3f3f3f3f;
const int maxn=;
int Map[maxn][maxn], f[maxn]; int main()
{
//freopen("in.txt", "r", stdin);
cin>>n;
for(int i=; i<=n; i++)
for(int j=; j<=i; j++)
cin>>Map[i][j]; int ans=-INF;
for(int i=; i<=n; i++)
for(int j=i; j>; j--)
{
f[j]=max(f[j], f[j-])+Map[i][j];
if(i==n)
ans=max(ans, f[j]);
} cout<<ans<<endl;
return ;
}
很相似的一道题目,但搞了好久。
请保证Chair消灭最多敌人并且冲出包围圈。假设Chair在左上角,达到右下角时算突围成功。
值得注意的是,Chair是如此英勇,应当一往无前,也就是说只能向下或向右走。
图示应为最大战果——于是你应该输出他们的和:35
#include <bits/stdc++.h>
using namespace std ;
int main ()
{
long long dp[][];
//int T; cin>>T;
//while (T--)
{
int n; cin>>n;
memset(dp,,sizeof(dp));
for(int i=; i<n; i++)
for(int j=; j<n; j++)
cin>>dp[i][j];
/*
for(int i=n-1; i>=0; i--)
for(int j=0; j<n; j++)
dp[i][j]+=max(dp[i+1][j],dp[i][j+1]);
*/ //错解,重叠子结构顺序弄反了; for(int i=n-; i>=; i--)
for(int j=n-; j>=; j--)
dp[i][j]+=max(dp[i+][j],dp[i][j+]); /*
for(int i=0; i<n; i++)
{
for(int j=0; j<n; j++)
cout<<dp[i][j]<<" ";
cout<<endl;
}
*/
cout<<dp[][]<<endl; }
return ;
}
今天又遇到一个很相似的题目,a了好久,发现自己跟个智障一样。