码迷,mamicode.com
首页 > 其他好文 > 详细

Codeforces 825D 二分贪心

时间:2017-07-27 18:35:48      阅读:101      评论:0      收藏:0      [点我收藏+]

标签:i++   nts   scan   防止   stream   force   nbsp   puts   codeforce   

题意:给一个 s 串和 t 串, s 串中有若干问号,问如何填充问号使得 s 串中字母可以组成最多的 t 串。输出填充后的 s 串。

思路:想了下感觉直接怼有点麻烦,要分情况:先处理已经可以组成 t 串的部分,然后处理 s 串中可以利用的部分,如果还有问号剩余,再直接怼。但如果用二分写就很直观了,直接看最多能组成多少个 t 串。

居然踩了long long的坑感觉自己宛若一个zz。check函数中的计算要用long long防止溢出= =(明明大水题的说。

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
typedef long long ll;
const int N=1e6+10;
char s[N],t[N];
int cntS[30],cntT[30],cntQ;
bool check(int x){
    ll tmp=0;
    for(int i=0;i<26;i++) if(cntT[i]){
        tmp+=max(0LL,1LL*x*cntT[i]-cntS[i]);
    }
    return tmp<=cntQ;
}
int main(){
    scanf("%s%s",s,t);
    int ls=strlen(s),lt=strlen(t);
    for(int i=0;i<ls;i++)
        if(isalpha(s[i]))
            cntS[s[i]-a]++;
        else cntQ++;
    for(int i=0;i<lt;i++)
        cntT[t[i]-a]++;
    int l=0,r=N,x=0;
    while(l<=r){
        int mid=(l+r)>>1;
        if(check(mid)){
            l=mid+1;
            x=mid;
        }else r=mid-1;
    }
    int id=0;
    for(int i=0;i<ls;i++){
        if(s[i]==?){
            while(id<26&&cntS[id]>=cntT[id]*x) id++;
            if(id<26){
                printf("%c",a+id);
                cntS[id]++;
            }
            else putchar(a);
        }else printf("%c",s[i]);
    }
    puts("");
    return 0;
}

 

Codeforces 825D 二分贪心

标签:i++   nts   scan   防止   stream   force   nbsp   puts   codeforce   

原文地址:http://www.cnblogs.com/names-yc/p/7246018.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!