uva10635 【LIS】

题意:求两个数组的最长公共子序列长度(LCS),每个序列中的元素没有重复的且取值范围1~n*n。

分析:直接求LCS时间复杂度为o(n*n)肯定不行,考虑题目的特殊性,数的范围1~n*n,且任意两个数都不相同。

如果把第一个数组重新编号:12,3,4...p+1。把第二个数组也对应起来,实际上问题就转化为了求第二个数组的LIS(可以用o(nlgn)的算法求解。问题就得到了解决。

代码:

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<string>
#include<vector>
#include<queue>
#include<cmath>
#include<stack>
#include<set>
#include<map>
#define INF 0x3f3f3f3f
#define Mn 62510
#define Mm 2000005
#define mod 1000000007
#define CLR(a,b) memset((a),(b),sizeof((a)))
#define CPY(a,b) memcpy ((a), (b), sizeof((a)))
#pragma comment(linker, "/STACK:102400000,102400000")
#define ul u<<1
#define ur (u<<1)|1
using namespace std;
typedef long long ll;
int x;
int a[Mn],b[Mn],d[Mn];
int main() {
    int T;
    scanf("%d",&T);
    int n,p,q,x;
    for(int cas=1;cas<=T;cas++){
        scanf("%d%d%d",&n,&p,&q);
        CLR(a,0);
        for(int i=1;i<=p+1;i++) {
            scanf("%d",&x);
            a[x]=i;
        }
        int cnt=0;
        for(int i=1;i<=q+1;i++) {
            scanf("%d",&x);
            if(a[x]) b[cnt++]=a[x];
        }
        int top=1;CLR(d,-1);
        int ans=0;
        for(int i=0;i<cnt;i++) {
            int pos=lower_bound(d,d+top,b[i])-d;
            ans=max(pos,ans);
            if(d[pos]==-1) top++;
            d[pos]=b[i];
        }
        printf("Case %d: %d\n",cas,ans);
    }
    return 0;
}


你可能感兴趣的:(uva10635 【LIS】)