传送门
题意:
现在有一个
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
n−1个数
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;
}