Codeforces 794C【贪心】

一开始以为计数n*26的算法很优秀哦?后面确实时间还是优秀的,只不过后面wa的自己代码就越来越浪,难看的…..;
对于1串只会取前(n+1)/2个字符,对于2串只会取n/2个字符(都是整除),然后每次拿1串的最小和2串的最大比一比,对于填1串,如果最小<最大,字典序越小,小的放最前,不放最前会被挤掉,如果最小>最大,字典序越小,大的放后面,然后2串的更大的放前面;对于填2串,如果最小<最大,字典序越大,大的放最前,不放最前会被挤掉,如果最小>最大,字典序越大,小的放后面,然后2串的更大的放前面

#include<iostream>
#include<cstdio>
#include<queue>
#include<vector>
#include<string.h>
#include<algorithm>
using namespace std;
typedef long long LL;
#define mem(a,b) memset(a, b, sizeof(a))
#define Lson num<<1,Left,Mid
#define Rson num<<1|1, Mid+1, Right

const int Maxn = 3e5 + 10;

char s1[Maxn], s2[Maxn], ans[Maxn];
int n, x, num, cnt1[30],cnt2[30], Left, Right, sum;
int u, v;
bool flag;
bool Judge(){
    for(int i=0;i<26;i++){
        if(cnt1[i]){
            u = i;
            break;
        }
    }
    for(int i=25;i>=0;i--){
        if(cnt2[i]){
            v = i;
            break;
        }
    }
    return u < v;
}

int main(){
    scanf("%s%s",s1,s2);
    n = strlen(s1);
    mem(cnt1, 0);
    mem(cnt2, 0);
    for(int i=0;i<n;i++){
        x = s1[i] - 'a', cnt1[x]++;
        x = s2[i] - 'a', cnt2[x]++;
    }
    sum = 0;
    for(int i=0;i<26;i++){
        if(sum + cnt1[i] >= (n+1)/2){
            cnt1[i] = (n+1)/2 - sum;
            for(int j=i+1;j<26;j++) cnt1[j] = 0;
            break;
        }
        sum = sum + cnt1[i];
    }
    sum = 0;
    for(int i=25;i>=0;i--){
        if(sum + cnt2[i] > n/2){
            cnt2[i] = n/2 - sum;
            for(int j=i-1;j>=0;j--) cnt2[j] = 0;
            break;
        }
        sum = sum + cnt2[i];
    }

    num = 0;
    Left = 0, Right = n - 1;
    while(num < n){
        if(Judge()){
            for(int i=0;i<26;i++){
                if(cnt1[i]){
                    u = i;
                    cnt1[i]--;
                    break;
                }
            }
            ans[Left++] = u + 'a';
            num++;
            if(num == n) break;
        }
        else{
            for(int i=25;i>=0;i--){
                if(cnt1[i]){
                    v = i;
                    cnt1[i]--;
                    break;
                }
            }
            ans[Right--] = v + 'a';
            num++;
            if(num == n) break;
        }
        if(Judge()){
            for(int i=25;i>=0;i--){
                if(cnt2[i]){
                    v = i;
                    cnt2[i]--;
                    break;
                }
            }
            ans[Left++] = v + 'a';
            num++;
            if(num == n) break;
        }
        else{
            for(int i=0;i<26;i++){
                if(cnt2[i]){
                    u = i;
                    cnt2[i]--;
                    break;
                }
            }
            ans[Right--] = u + 'a';
            num++;
            if(num == n) break;
        }
    }
    ans[num] = '\0';
    printf("%s\n",ans);
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值