首页 > 其他 > 详细

CF1138D.Camp Schedule

时间:2019-03-09 11:15:44      阅读:233      评论:0      收藏:0      [点我收藏+]

传送门

虽然是D,但是还是Sb题,把模式串跑一遍KMP,然后把按顺序放,每次放完之后跳到对应的前缀,继续放。
如果最后1的数量还有剩,再将最后的位数全部放1
代码:

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
using namespace std;
void read(int &x) {
    char ch; bool ok;
    for(ok=0,ch=getchar(); !isdigit(ch); ch=getchar()) if(ch=='-') ok=1;
    for(x=0; isdigit(ch); x=x*10+ch-'0',ch=getchar()); if(ok) x=-x;
}
#define rg register
const int maxn=5e5+10;
char p[maxn],s[maxn];
int num,num1,len,len1,nxt[maxn],ans;
int main()
{
    scanf("%s",p+1),scanf("%s",s+1);
    len=strlen(p+1),len1=strlen(s+1);
    for(rg int i=1;i<=len;i++)if(p[i]=='1')num++;
    for(rg int i=1;i<=len1;i++)if(s[i]=='1')num1++;
    if(num<num1||len<len1)
    {
        for(rg int i=1;i<=len;i++)printf("%c",p[i]);
        printf("\n");return 0;
    }
    for(rg int i=2,j=0;i<=len1;i++)
    {
        while(j&&s[j+1]!=s[i])j=nxt[j];
        if(s[j+1]==s[i])j++;
        nxt[i]=j;
    }
    ans=len+1;
    for(rg int i=1,j=0;i<=len;i++)
    {
        if(num&&s[j+1]=='1')num--,p[i]='1',j++;
        else if(s[j+1]=='0')j++,p[i]='0';
        else {ans=i;break;}
        if(j==len1)j=nxt[j];
    }
    for(rg int i=ans;i<=len;i++)p[i]='0';
    while(num)
    {
        ans--;
        if(p[ans]=='1')num++;
        num--,p[ans]='1';
    }
    for(rg int i=1;i<=len;i++)printf("%c",p[i]);
}

CF1138D.Camp Schedule

原文:https://www.cnblogs.com/lcxer/p/10499893.html

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