bzoj3325: [Scoi2013]密码(manacher+模拟)

传送门
题意:
现在有一个 n n n个小写字母组成的字符串 s s s
然后给你 n n n个数 a i a_i ai, a i a_i ai表示以 s i s_i si为中心的最长回文串串长。
再给你 n − 1 n-1 n1个数 b i b_i bi, b i b_i bi表示以 s i , s i + 1 s_i,s_{i+1} si,si+1中间空隙为中心的最长回文串串长。
问你满足条件的所有 s s s串中字典序最小的那个。


思路:
可以根据 a i a_i ai b i b_i bi逆模拟 m a n a c h e r manacher manacher的过程贪心构造这个 s s s
代码:

#include<bits/stdc++.h>
#define ri register int
using namespace std;
const int rlen=1<<18|1;
inline char gc(){
    static char buf[rlen],*ib,*ob;
    (ib==ob)&&(ob=(ib=buf)+fread(buf,1,rlen,stdin));
    return ib==ob?-1:*ib++;
}
inline int read(){
    int ans=0;
    char ch=gc();
    while(!isdigit(ch))ch=gc();
    while(isdigit(ch))ans=((ans<<2)+ans<<1)+(ch^48),ch=gc();
    return ans;
}
const int N=2e5+5;
bool ban[N][26];
int n,col[N],r[N],mx=0,id=0;
int main(){
    n=read(),r[1]=1;
    for(ri i=1,j=2;i<=n;++i,j+=2)r[j]=read()+1,col[j]=-1;
    for(ri i=1,j=3;i<n;++i,j+=2)r[j]=read()+1,col[j]=-1;
    for(ri i=1,pos;i<=n<<1;++i){
        if(i&1)col[i]=26;
        else if(col[i]==-1)for(ri j=0;j<26;++j)if(!ban[i][j]){col[i]=j;break;}
        pos=id+mx>i?min(id+mx-i,r[id*2-i]):1;
        while(pos<r[i])col[i+pos]=col[i-pos],++pos;
        if(i-r[i]>1&&(i-r[i])%2==0)ban[i+r[i]][col[i-r[i]]]=1;
        if(i+r[i]>mx)mx=r[i],id=i;
        if(!(i&1))cout<<(char)(col[i]+'a');
    }
    return 0;
}
相关推荐
©️2020 CSDN 皮肤主题: 创作都市 设计师:CSDN官方博客 返回首页