P1368 工艺 SA/最小表示法

正解:SA/最小表示法

解题报告:

传送门!

听说正解是最小表示法,,,O(n)然后常数还挺小的,,,

但是我不会QAQ!

所以先写下SA的做法趴,,,等get了最小表示法再来写正解QAQ

就这种题算是比较套路的,,,首先看到是个环,显然就考虑到倍增?

然后就跑个SA

然后从小往大扫sa数组,如果当前位置是<=序列长度n的,就直接输出然后return就好

注意一下的是n因为倍增了,所以长度要*2,这里建议是直接*2然后在求完之后再/2,不然每次都要打n*2的很容易错,,,

over,放下代码就好QAQ

#include<bits/stdc++.h>
using namespace std;
#define il inline
#define gc getchar()
#define ri register int
#define rc register char
#define rb register bool
#define rp(i,x,y) for(ri i=x;i<=y;++i)
#define my(i,x,y) for(ri i=x;i>=y;--i) const int N=+;
int n,t[N<<],sa[N<<],ch[N<<],str[N<<],y[N<<],x[N<<]; il int read()
{
rc ch=gc;ri x=;rb y=;
while(ch!='-' && (ch>'' || ch<''))ch=gc;
if(ch=='-')ch=gc,y=;
while(ch>='' && ch<='')x=(x<<)+(x<<)+(ch^''),ch=gc;
return y?x:-x;
}
il bool cmp(ri gd,ri gs,ri k){return y[gd]==y[gs] && y[gd+k]==y[gs+k];}
il void SA()
{
ri m=n;
rp(i,,n)++t[x[i]=str[i]];
rp(i,,m)t[i]+=t[i-];
my(i,n,)sa[t[x[i]]--]=i;
for(ri k=;k<=n;k<<=)
{
ri p=;
rp(i,,n)y[i]=;rp(i,,m)t[i]=;
rp(i,n-k+,n)y[++p]=i;rp(i,,n)if(sa[i]>k)y[++p]=sa[i]-k;
rp(i,,n)++t[x[y[i]]];
rp(i,,m)t[i]+=t[i-];
my(i,n,)sa[t[x[y[i]]]--]=y[i];
swap(x,y);
x[sa[]]=p=;
rp(i,,n)x[sa[i]]=cmp(sa[i],sa[i-],k)?p:++p;
if(p>=n)break;m=p;
}
} int main()
{
// freopen(".in","r",stdin);freopen(".out","w",stdout);
n=read();rp(i,,n)str[i]=str[i+n]=read();n<<=;SA();//rp(i,1,n)if((sa[i]<<1)<=n)printf("%c",str[sa[i]+n-1]);
rp(i,,n)
if((sa[i]<<)<=n)
{
rp(j,,(n>>)-)printf("%d ",str[sa[i]+j]);return ;
}
return ;
}

放下代码QwQ

上一篇:mybatis-generator生成model和dao层代码


下一篇:安装redis报错 you need tcl 8.5 or newer in order to run redis test