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

这篇博客介绍了如何解决[Scoi2013]密码问题,通过manacher算法结合给定的回文串长度信息,逆向贪心构造字典序最小的字符串。内容包括题目描述、解题思路和实现代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

传送门
题意:
现在有一个 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;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值