1010. 拦截导弹

某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统。

但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度。

某天,雷达捕捉到敌国的导弹来袭。

由于该系统还在试用阶段,所以只有一套系统,因此有可能不能拦截所有的导弹。

输入导弹依次飞来的高度(雷达给出的高度数据是不大于30000的正整数,导弹数不超过1000),计算这套系统最多能拦截多少导弹,如果要拦截所有导弹最少要配备多少套这种导弹拦截系统。

输入格式

共一行,输入导弹依次飞来的高度。

输出格式

第一行包含一个整数,表示最多能拦截的导弹数。

第二行包含一个整数,表示要拦截所有导弹最少要配备的系统数。

输入样例:

389 207 155 300 299 170 158 65

输出样例:

6
2

分析:
这道题的题意就是,找单调下降最长的长度,然后如果第一遍找不完,再从剩下的导弹高度里边在进找,直到找到所有严格单调下降的导弹高度,并统计找的次数

换句人话--->用多少个单调下降子序列把完整的序列覆盖掉

贪心的流程:
  从前往后扫描每个数:
    case 1:如果现有的子序列都小于当前的数,就创建一个新的子序列
    case 2:将当前数放在大于等于它的最小子序列后面

1010. 拦截导弹_第1张图片

 

1010. 拦截导弹_第2张图片

这个g[]数组 就相当于我们的子序列末尾的数,他要保证单调上升,而我们的a就是当前子序列末尾的大于等于x的最小值,也就是将x放在了子序列结尾大于等于x的子序列的后面,也就是a的后边;

这道题如果深挖的化有一个反链的叫做DailWorth定理,有兴趣的可以了解下,在这里我就不赘述了。

这里的输入比较特别可以参考下c++的stringstream

 

具体详解代码:

 1 #include
 2 #include
 3 
 4 using namespace std;
 5 const int N = 1010;
 6 int n;
 7 int q[N];
 8 int f[N],g[N];
 9 
10 int main(){
11     while(cin >> q[n]) n++;
12     int res = 0;
13     for(int i = 0;i < n;i++){
14         f[i] = 1;
15         for(int j = 0;j < i;j++)
16             if(q[j] >= q[i]) f[i] = max(f[i],f[j] + 1);
17             res = max(res,f[i]);
18     }
19     cout << res << endl;
20     
21     //用g[]数组存储我们当前现有的所有的子序列
22     //用cnt表示我们当前现有子序列的个数
23     int cnt  = 0;
24     //从前往后贪心一遍
25     for(int i = 0;i < n;i++){
26         int k = 0;//k表示我们从前往后找的序列
27         //只要我们没有遍历完所有的序列 并且 当前的序列的结尾小于我们当前的数,就k++
28         while(k < cnt && g[k] < q[i]) k++;//序列找完并且没有比序列最小值小的我们的序列就++
29         g[k] = q[i];//将大的q[i]交换到本来小的序列的位置上在进行判断
30         if(k >= cnt) //说明我们没有任何一个序列是大于我们的当前数的
31         //我们就开一个新的序列
32         cnt++;
33     } 
34     
35     cout << cnt;
36     
37 }

 

 

 





你可能感兴趣的:(1010. 拦截导弹)