动态规划-拦截导弹
某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统。但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度。某天,雷达捕捉到敌国的导弹来袭。由于该系统还在试用阶段,所以如果只有一套系统,有可能不能拦截所有的导弹。
输入导弹依次飞来的高度(雷达给出的高度数据是不大于30000的正整数),计算这套系统最多能拦截多少导弹,如果要拦截所有导弹最少要配备多少套这种导弹拦截系统。
【输入描述】
输入2行。第一行导弹的数量,第二行导弹的高度。
【输出描述】
输出2行。第一行一套系统拦截最多导弹数量,第二行系统数量。
【输入样例】
8
389 207 155 300 299 170 158 65
【输出样例】
6
2
【参考程序】
//小牛编程
#include <bits/stdc++.h>
using namespace std;
const int MAXN = 1001;
// f:最长不上升子序列计数,g:系统数量计数,p:不上升子序列的下标(路径)
int a[MAXN], f[MAXN], g[MAXN], p[MAXN];
int main() {
int n = 1, ans = 0, cnt = 0, ansIndex = 1;
cin >> n;
for (int i = 1; i <= n; i++)
cin >> a[i];
for (int i = n; i >= 1; i--) {
p[i] = i;
for (int j = i + 1; j <= n; j++) {
if (a[j] <= a[i]) {
f[i] = max(f[i], f[j]);
if (f[j] + 1 > f[i]) { //最长子序列加入路径中
p[i] = j;
}
} else {
g[i] = max(g[i], g[j]);
}
}
++f[i];
if (f[i] >= ans) {
ans = f[i];
ansIndex = i;
}
// ans=max(ans,++f[i]);
cnt = max(cnt, ++g[i]);
}
cout << ans << endl << cnt << endl;
// 补充:输出最多拦截到的导弹序列
int t = ansIndex;
for (int i = 1; i <= ans; i++) {
cout << a[t] << " ";
t = p[t];
}
return 0;
}