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

本文介绍了一种利用Manacher算法逆向构造字符串的方法,该方法能够找到满足特定条件的字典序最小的回文串。通过分析给定的字符中心及其间的最长回文长度,算法能够逐步确定字符串中每个位置的字符。

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

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


思路:
可以根据aia_iaibib_ibi逆模拟manachermanachermanacher的过程贪心构造这个sss
代码:

#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;
}

转载于:https://www.cnblogs.com/ldxcaicai/p/10633602.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值