最长子序列问题

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

输入导弹依次飞来的高度(雷达给出的高度数据是不大于50000的正整数),(1)计算这套系统最多能拦截多少导弹
(2)如果要拦截所有导弹最少要配备多少套这种导弹拦截系统。
【解析】:此题的第一问即是求最长不上升子序列的,第二问是求最长上升子序列。
一.最长不上升子序列
首先,从前往后定义子序列,当后面的不大于前面的时候,更新子序列长度和子序列端点,如果后面的大于前面的时候,二分查找,更新长度为最长且小于数值小于自己的不上升序列的结束点。
核心代码(必背)

f[0]=1234123412;//这个数要大于50000,不然可能你就无法更新
   int ans=0;//最长不上升子序列的长度
   for(int i=1;i<=n;i++){
          if(a[i]<=f[ans]){
                f[ans1+1]=a[i];//结束点为a[i]
                ans1++; //当前最长不上升序列的长度加一
       }
       else {//二分查找
              l=0;r=ans1;
              while(l<r){//当查找区间不为空
                     mid=(l+r)/2;//查找区间正中的下标
                    if(a[i]<f[mid])l=mid+1;
                    else {
                        r=mid;    
              }
           }
           if(l!=0)f[l]=a[i];
       }
   }

二。最长上升子序列
同理,把最长不上升子序列逆过来就是此答案了
核心代码:

f[0]=1234123412;
 int ans=0;
for(int i=1;i<=n;i++){
          if(a[i]>f[ans]){
                f[ans+1]=a[i];//结束点为a[i]
                ans++; //当前最长上升序列长度加一
       }
       else {//二分查找
              l=0;r=ans;
              while(l<r){
                     mid=(l+r)/2;
                    if(a[i]>=f[ans])l=mid+1;
                    else {
                        r=mid;    
              }
           }
          if(l!=0)f[l]=a[i];
       }
   }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值