zoukankan      html  css  js  c++  java
  • bzoj 4698: Sdoi2008 Sandy的卡片【SAM】

    差分之后用SAM求LCS,然后答案就是LCS+1

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    using namespace std;
    const int N=2005;
    int n,m,s[N],b[N],fa[N],ch[N][205],dis[N],cur=1,con=1,la,f[N][N],c[N],a[N],ans;
    int read()
    {
    	int r=0,f=1;
    	char p=getchar();
    	while(p>'9'||p<'0')
    	{
    		if(p=='-')
    			f=-1;
    		p=getchar();
    	}
    	while(p>='0'&&p<='9')
    	{
    		r=r*10+p-48;
    		p=getchar();
    	}
    	return r*f;
    }
    void ins(int c,int id)
    {
        la=cur,dis[cur=++con]=id;
        int p=la;
        for(;p&&!ch[p][c];p=fa[p])
            ch[p][c]=cur;
        if(!p)
            fa[cur]=1;
        else
        {
            int q=ch[p][c];
            if(dis[q]==dis[p]+1)
                fa[cur]=q;
            else
            {
                int nq=++con;
                dis[nq]=dis[p]+1;
                memcpy(ch[nq],ch[q],sizeof(ch[q]));
                fa[nq]=fa[q];
                fa[q]=fa[cur]=nq;
                for(;ch[p][c]==q;p=fa[p])
                    ch[p][c]=nq;
            }
        }
    }
    int main()
    {
    	m=read(),n=read();
    	for(int i=1;i<=n;i++)
    		s[i]=read();
        for(int i=1;i<n;i++)
            ins(s[i+1]-s[i]+101,i);
        for(int i=1;i<=con;i++)
            f[i][1]=dis[i];
        for(int j=2;j<=m;j++)
        {
            n=read();
    		for(int i=1;i<=n;i++)
    			s[i]=read();
            for(int i=1,p=1,len=0;i<n;i++)
            {
                int c=s[i+1]-s[i]+101;
                if(ch[p][c])
                    len++,p=ch[p][c],f[p][j]=max(f[p][j],len);
                else
                {
                    for(;p&&!ch[p][c];p=fa[p]);
                    if(!p)
                        p=1,len=0;
                    else
                        len=dis[p]+1,p=ch[p][c],f[p][j]=max(f[p][j],len);
                }
            }
        }
        for(int i=1;i<=con;i++)
            c[dis[i]]++;
        for(int i=1;i<=con;i++)
            c[i]+=c[i-1];
        for(int i=1;i<=con;i++)
            a[c[dis[i]]--]=i;
        for(int j=2;j<=m;j++)
            for(int i=con;i>=1;i--)
                f[fa[a[i]]][j]=max(f[fa[a[i]]][j],f[a[i]][j]);
        for(int i=1;i<=con;i++)
        {
            for(int j=2;j<=m;j++)
                f[i][1]=min(f[i][1],f[i][j]);
            ans=max(ans,f[i][1]);
        }
        printf("%d
    ",ans+1);
        return 0;
    }
    
  • 相关阅读:
    HNOI2008玩具装箱
    CEOI2004锯木厂选址
    APIO2010特别行动队
    【HNOI2011】数学作业
    【JLOI2015】城池攻占
    魔法猪学院
    Kruskal重构树(货车运输)
    旋转卡壳求凸包直径
    Graham凸包算法简介
    poj-3169Layout
  • 原文地址:https://www.cnblogs.com/lokiii/p/10008609.html
Copyright © 2011-2022 走看看