线性dp
用于找最优解
例题
# P1020 [NOIP 1999 提高组] 导弹拦截
题目描述
某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统。但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度。某天,雷达捕捉到敌国的导弹来袭。由于该系统还在试用阶段,所以只有一套系统,因此有可能不能拦截所有的导弹。
输入导弹依次飞来的高度,计算这套系统最多能拦截多少导弹,如果要拦截所有导弹最少要配备多少套这种导弹拦截系统。
输入格式
一行,若干个整数,中间由空格隔开。
输出格式
两行,每行一个整数,第一个数字表示这套系统最多能拦截多少导弹,第二个数字表示如果要拦截所有导弹最少要配备多少套这种导弹拦截系统。
输入输出样例 #1
输入 #1
389 207 155 300 299 170 158 65
输出 #1
6
2
NOIP1999 提高组 第一题
思路(Dilworth定理):
最多能拦截导弹:最长不上升子序列
最少系统:最长上升子序列——B3637
代码
#include<bits/stdc++.h>
using namespace std;
int n,s[100005],a[100005]={0},b[100005],ans,len=0;
int main()
{
// freopen("in.in","r",stdin);
int i=1;
while(cin>>s[i]) i++;
i--;
for (int j=1;j<=i;j++)
{
if(a[a[0]]<s[j])
{
a[0]++;
a[a[0]]=s[j];
}
else
a[lower_bound(a+1,a+1+a[0],s[j])-a]=s[j];
}
for (int j=i;j>=1;j--)
if(b[len]<=s[j])
{
len++;
b[len]=s[j];
}
else
b[upper_bound(b+1,b+1+len,s[j])-b]=s[j];
cout<<len<<endl<<a[0];
return 0;
}

浙公网安备 33010602011771号