球的序列

【题目描述】:

N个编号为1-n的球,每个球都有唯一的编号。这些球被排成两种序列,分别为A、B序列,现在需要重新寻找一个球的序列l,对于这个子序列l中任意的两个球,要求j,k(j<k),都要求满足lj在A中位置比lk在A中位置靠前,且lj在B中位置比lk在B中位置靠前,请你计算这个子序列l的最大长度。

【输入描述】:

第一行一个整数,表示N。

第二行N个整数,表示A序列。

第三行N个整数,表示B序列。

【输出描述】:

一个数子序列l的最大长度。

【样例输入】:

5
1 2 4 3 5
5 2 3 4 1

【样例输出】:

2

【样例说明】:

L可以是{2,3},也可以是{2,4}

【时间限制、数据范围及描述】:

时间:1s 空间:256M

40% N<=5000

100% N<=50000

一看到题目就觉得巨坑(事实证明确实如此。。。),明显没思路。。。

后来经过题解,知道应该先预处理 B数组 中元素在 A数组 中的位置,再跑 LIS 。然而看了下数据范围,只有40分。。。

于是想起 LIS 的优化,即用一个数组保存长度为数组下标的子序列中元素的最小值。。。调试卡了我半个小时。。。

于是就这样了。。。

#include<iostream>
#include<algorithm>
#include<cstdio>
#define MAXN 50010
using namespace std;
int n,now=1;
int a[MAXN],b[MAXN],c[MAXN],num[MAXN],best[MAXN],f[MAXN];
inline int read(){
       int date=0,w=1;char c=0;
       while(c!='-'&&(c<'0'||c>'9'))c=getchar();
       if(c=='-'){w=-1;c=getchar();}
       while(c>='0'&&c<='9'){date=date*10+c-'0';c=getchar();}
       return date*w;
}
int main(){
    int l,r,mid;
    n=read();
    for(int i=1;i<=n;i++)a[i]=read();
    for(int i=1;i<=n;i++){
            b[i]=read();
            c[b[i]]=i;
            }
    for(int i=1;i<=n;i++)num[i]=c[a[i]];
    f[1]=1;
    best[1]=num[1];
    for(int i=2;i<=n;i++){
            l=0;
            r=now;
            while(l+1<r){
                         mid=l+r>>1;
                         if(best[mid]>=num[i])r=mid;
                         else l=mid;
                         }
            if(best[r]<num[i])
            if(r==now){
                       now++;
                       f[i]=now;
                       best[now]=num[i];
                       }
            else{
                 f[i]=r+1;
                 best[r+1]=min(best[r+1],num[i]);
                 }
            else{
                 f[i]=l+1;
                 best[l+1]=min(best[l+1],num[i]);
                 }
            }
    printf("%d\n",now);
    return 0;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值