初识动态规划-最长不上升子序列

题目数据来自 [NOIP]拦截导弹

 

题目描述

某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统。但是这种导弹拦截 系统有一个缺陷:虽然它的第一发炮弹能够达到任意的高度,但是以后每一发炮弹都不能高于前一发的高度。某天,雷达捕捉到敌国的导弹来袭。由于该系统还在试 用阶段,所以只有一套系统,因此有可能不能拦截所有的导弹。

输入

最多20个整数,分别表示导弹依次飞来的高度(雷达给出高度数据是不大于30000的正整数)

输出

第一行输出最多能拦截的导弹数
第二行输出要全部拦截最少需要的拦截系统数

样例输入

389 207 155 300 299 170 158 65

样例输出

6
2

 

第一次做到这一类的题目,完全不知道从何做起。参考网上代码并加上自己的理解之后写下代码存下来。相信读者朋友们在亲自用笔一步步计算之后会和我一样理解最长不上升子序列算法其中的巧妙之处!

第一步,我们先要知道什么是“最长不上升子序列”。

以测试用例为389 207 155 300 299 170 158 65为例。所谓最长不上升子序列,顾名思义,就是降序排列的数字最长的一个序列,测试用例的最长不上升子序列为389 300 299 170 158 65。所以这组实例的最长不上升子序列的长度为6。

备注:这不是AC代码!!!

一定在纸上演算一遍算法过程,就以样例为例,演算一遍之后就能加深对算法的理解!!!

 

#include 
int a[1010];
int la[1010];
int main()
{
    int i,j,n,max,max1=1;
    scanf("%d",&n);
    
    for(i=1;i<=n;i++)
        la[i]=1; //la数组存以第i个数为终点的最长不上升序列,数组la初始化为1,因为数最少都有一位,所以最长不上升子序列最少也为1

    for(i=1;i<=n;i++)
        scanf("%d",&a[i]);//a数组保存序列数组
    
    for (i=2;i<=n;i++)
    {
        max=0;//max存的是以a[j]为终点的最长不上升子序列。而且需要注意的是,每一次新循环的开始,max进行了初始化,而la数组没有初始化,一直存的是以a[i]为终点的最长不上升子序列。
        for (j=1;j<=i-1;j++)
            if(a[j]>a[i]&&la[j]>max)  //在前i-1个序列中,寻找以a[j]为终点大于a[i]的最长的子序列,并更新max值
                max=la[j];

        la[i]=max+1;因为j最大只能到i-1,所以以a[i]为终点的最长不上升子序列肯定比以a[j]为终点的最长不上升子序列要长1.
        if(la[i]>max1)           //对于这一段的理解,如果找到了比a[i]大的数,并且la[j](第i个数为终点的最长不上升序列)比max更大,则更新max的值,并将最后的max存入la[i]
            max1=la[i];
    }
    printf("%d\n",max1);
    return 0;
}

 

对la数组的理解很重要,读者在读代码的同时需要牢记la数组存的是什么。

 

 

你可能感兴趣的:(初识动态规划-最长不上升子序列)