Codeforces 825D 二分贪心

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Codeforces 825D 二分贪心相关的知识,希望对你有一定的参考价值。

题意:给一个 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 二分贪心的主要内容,如果未能解决你的问题,请参考以下文章

[CF825D] Suitable Replacement (贪心乱搞)

Codeforces 830A. Office Keys (背包dp+贪心) / (二分+贪心)

codeforces 1324 C. Frog Jumps(贪心/二分)

CodeForces - 1251D (贪心+二分)

CodeForces - 589A(二分+贪心)

CodeForces 343C Read Time 二分答案+贪心