【题目描述】:
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;
}