传送门: HDU_1087 http://acm.hdu.edu.cn/showproblem.PHP?pid=1087
题意: 现在要玩一个跳棋类游戏, 有棋盘和棋子. 从棋子 st 开始, 跳到棋子 en 结束. 跳动棋子的规则是下一个落脚的棋子的号码必须要大于当前棋子的号码. st 的号是所有棋子中最小的, en 的号是所有棋子中最大的. 最终所得分数是所有经过的棋子的号码的和.
思路: 读完题之后知道这是一个最长上升子序列的题目. 因为之前刚刚看过牛客网上一节讲解最长上升子序列的视屏, 所以一上来就找准了方向, but 我只知道怎么求最长上升子序列的长度啊, 和怎么求??? 于是自己想方法开始求和, 然后就 wa 掉了一个上午. 下午起床后, 搜了一下题解, 看过思路后发现这种动规的做法, 之前用到过, 但是,,,,,,,, 罪过罪过.
复杂度: O(n^2), 对每一个棋子分配一个 dp, 每个棋子, 遍历他之前的棋子并从中找出 dp[j]+a[i] 最大的值赋给 dp[i].
代码:
- #include <iostream>
- #include <queue>
- #include <cstdio>
- #include <algorithm>
- #include <cmath>
- #include <cstring>
- #define INF 0x3f3f3f3f
- using namespace std;
- typedef long long ll;
- const int maxn = 2010;
- ll a[maxn],b[maxn],dp[maxn];
- int main()
- {
- iOS::sync_with_stdio(false);
- int n;
- while(cin>>n && n)
- {
- memset(a,0,sizeof(a));
- memset(dp,0,sizeof(dp));
- for(int i = 0; i<n; i++)
- cin>>a[i];
- ll ans = -1e9;
- for(int i = 0; i<n; i++)
- {
- dp[i] = a[i];
- for(int j = 0; j<i; j++)
- {
- if(a[j]<a[i])
- {
- dp[i] = max(dp[j]+a[i], dp[i]);
- }
- }
- ans = max(ans,dp[i]);
- }
- cout<<ans<<endl;
- }
- return 0;
- }
- /*
- 样例输入:
- 3 1 3 2
- 4 1 2 3 4
- 4 3 3 2 1
- 0
- 样例输出:
- 4
- 10
- 3
- */
- View Code
来源: http://www.bubuko.com/infodetail-2790577.html