BZOJ 2882 工艺 ——后缀自动机 最小表示法

先说后缀自动机的做法。

直接把S串复制一遍成SS,然后建立后缀自动机,go边相当于在当前字符的后面插入,而son边可以看作在字符串前面加一个字符。

所以贪心的走字典序最小的边即可,而且根据后缀自动机的构建的性质,是一定可以走够len的,然后输出即可。

人生第一个后缀自动机的代码。

 

#include<iostream>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<map>
#include<cstdio>
#define N 600003
using namespace std;
int n,m,cnt,nq,np,p,q,last,root;
map<int,int> ch[N*2];
int fa[N*2],l[N*2],a[N];
void extend(int x)
{
    int c=a[x];
    p=last; np=++cnt; last=np;
    l[np]=x;
    for (;p&&!ch[p][c];p=fa[p]) ch[p][c]=np;
    if (!p) fa[np]=root;
    else {
        q=ch[p][c];
        if (l[q]==l[p]+1) fa[np]=q;
        else {
            nq=++cnt; l[nq]=l[q]+1;
            ch[nq]=ch[q];
            fa[nq]=fa[q];
            fa[q]=fa[np]=nq;
            for (;ch[p][c]==q;p=fa[p]) ch[p][c]=nq;
        }
    }
}
int main()
{
    scanf("%d",&n);
    for (int i=1;i<=n;i++) scanf("%d",&a[i]);
    for (int i=1;i<=n;i++) a[n+i]=a[i];
    m=2*n;
    last=root=++cnt; ch[N].clear();
    for (int i=1;i<=m;i++) extend(i);
    p=1;
    map<int,int>::iterator t;
    for (int i=1;i<=n;i++) {
        t=ch[p].begin();
        p=t->second;
        if (i!=n) printf("%d ",t->first);
        else printf("%d\n",t->first);
    }
}

然后再来说说最小表示法,大概的思想就是用两个指针来比较,然后不相等的时候就会排除很大的一片区间,然后最后剩下的就是最优解,最大表示法同理,写完后跑的飞快。

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define F(i,j,k) for (int i=j;i<=k;++i)
#define D(i,j,k) for (int i=j;i>=k;--i)
#define maxn 300005
int n,a[maxn],pos;
 
int main()
{
    scanf("%d",&n);
    F(i,1,n) scanf("%d",&a[i]);
    int i=1,j=2,k=0,flag=0;
    while (i<=n&&j<=n)
    {
        k=0;
//      printf("%d %d\n",i,j); 
        while (a[(i+k-1)%n+1]==a[(j+k-1)%n+1]&&k<=n) k++;
        if (k==n) {pos=i;flag=1;}
        else
        {
            if (a[(i+k-1)%n+1]>a[(j+k-1)%n+1])
            {
                if (i+k+1<=j) i=j+1;
                else i=i+k+1;
            }
            else
            {
                if (j+k+1<=i) j=i+1;
                else j=j+k+1;
            }
        }
    }
    if (!flag) pos=min(i,j);
    F(i,1,n) printf("%d%c",a[(pos+i-2)%n+1],i==n?'\n':' ');
}

  

 

posted @ 2017-03-01 22:34  SfailSth  阅读(382)  评论(0编辑  收藏  举报