首页 > 其他 > 详细

AGC037E - Reversing and Concatenating 题解

时间:2020-02-29 02:25:10      阅读:46      评论:0      收藏:0      [点我收藏+]

原题链接:E - Reversing and Concatenating
题目大意:给定一个字符串长度为\(N\)的字符串\(S\),每次将它和它的反串拼起来,然后从中截取一段长度为\(N\)的子串,并用这个字串覆盖原串,进行\(K\)次这样的操作后,使得所得串字典序最小。


题解:首先,我们可以发现,若令\(S\)串中的最小字符为\(c\),那么最终的串不可能比\(N\)\(c\)还要小。

然后,如果原串中\(c\)的连续长度最多为\(len\),那么每次一定会使\(c\)的连续串长度翻倍。

所以,\(K\)的数据范围是唬人的,当\(N \leq len\cdot 2^{k-1}\)是,答案就是\(N\)\(c\)

那么当\(N> len\cdot 2^{k-1}\)时该怎么办呢?\(N \leq 5000\),直接\(O(n^2)\)暴力上,查出一开始\(U\)的最小子串,然后输出\(len\cdot 2^{k-1}\)\(c\),再按着顺序输出后面的就可以了。

接下来是代码:

#include <cstdio>
const int Maxn=10000;
char s[Maxn+5];
int n,k;
char *ans;
bool check(char *a,char *b){
    for(int i=1;i<=n;i++){
        if(a[i]<b[i]){
            return 1;
        }
        if(a[i]>b[i]){
            return 0;
        }
    }
    return 0;
}
int main(){
    scanf("%d%d",&n,&k);
    scanf("%s",s+1);
    while(s[++n]!='\0');
    n--;
    for(int i=1;i<=n;i++){
        s[(n<<1)-i+1]=s[i];
    }
    ans=s;
    for(int i=1;i<=n;i++){
        if(check(s+i-1,ans)){
            ans=s+i-1;
        }
    }
    int len=1;
    while(len<=n&&ans[++len]==ans[1]);
    len--;
    if(k>=15||(len*(1<<(k-1))>=n)){
        for(int i=1;i<=n;i++){
            putchar(ans[1]);
        }
        puts("");
    }
    else{
        int now=n-(len*(1<<(k-1)));
        for(int i=1;i<=(len*(1<<(k-1)));i++){
            putchar(ans[1]);
        }
        for(int i=1;i<=now;i++){
            putchar(ans[i+len]);
        }
        puts("");
    }
    return 0;
}

AGC037E - Reversing and Concatenating 题解

原文:https://www.cnblogs.com/withhope/p/12380757.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!