• 51nod


    https://www.51nod.com/Challenge/Problem.html#!#problemId=1420
    一开始乱搞了一发,每个袋鼠二分找最小的能放它的,然后二分的范围从下一个开始保证不会把两个小袋鼠装在同一个里面,还过了一半的数据……

    然后才发现袋鼠并不能嵌套。想打vis标记大袋鼠跳过大袋鼠,然后样例都过不了。

    又想了半天网络流,流个鬼鬼流。

    看了一下别人的提示,贪心加二分。

    好像我误解了别人的贪心加二分,跑得比别人还快快。

    明显选的袋鼠一定是最小的那波,这样最优。

    然后二分选的袋鼠的数量,套在我第一次交的那个里面,一发就过了,500ms,应该是正解的做法。

    #include <bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    
    int n;
    int a[500005];
    int cpa[500005];
    
    bool check(int m) {
        int be=m,en=n;
        for(int i=0; i<m; i++) {
            int res=lower_bound(a+be,a+en,2*a[i])-a;
            if(res==en)
                return false;
            else {
                be=res+1;
            }
        }
        return true;
    }
    
    int main() {
        while(~scanf("%d",&n)) {
            for(int i=0; i<n; i++) {
                scanf("%d",&a[i]);
            }
            sort(a,a+n);
            for(int i=0;i<m;i++){
    
            }
    
            int ans=0;
            int l=0,r=n/2;
            while(1) {
                int m=(l+r)>>1;
                if(m==l){
                    if(check(r)){
                        ans=n-r;
                    }
                    else{
                        ans=n-l;
                    }
                    break;
                }
                if(check(m)){
                    l=m;
                }
                else{
                    r=m-1;
                }
            }
            printf("%d
    ",ans);
        }
    }
    
    

    然后看见一个更惊人的,从每个袋鼠开始选最大的能放进它的。好像没什么问题,这个贪心策略是对的。

    2,4,5,8
    从小的开始贪心,错在假如2用了4,虽然减少了一个,但假如2用了5,4用了8,那就减少两个。因为使用了4会影响4不能选8,所以不能这样贪心。

    但是从大的开始贪心就没问题,例如8可以装4,可以装2,那装4一定更好。(对个鬼鬼,假如有3呢?把4吃了2放哪里?)。

    其实这个策略的正确性应该在于先分成前后两部分(至于为什么要把奇数多出来的放在后部分,其实特判cnt不能超过n/2也可以,不然就有可能嵌套),前部分从后部分中选最合适的。这样不会有嵌套。但居然也是500ms,感觉被骗了。

    #include <bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    
    int n;
    int a[500005];
    int main() {
        while(~scanf("%d",&n)) {
            for(int i=0; i<n; i++) {
                scanf("%d",&a[i]);
            }
            sort(a,a+n,greater<int>());
    
            int cnt=0;
            for(int i=(n+1)/2;i<n;i++){
                if(a[cnt]>=2*a[i])
                    cnt++;
            }
    
            printf("%d
    ",n-cnt);
        }
    }
    

    #include <bits/stdc++.h>
    using namespace std;
    typedef long long ll;
    
    int n;
    int a[500005];
    int main() {
        while(~scanf("%d",&n)) {
            for(int i=0; i<n; i++) {
                scanf("%d",&a[i]);
            }
            sort(a,a+n,greater<int>());
    
            int cnt=0;
            for(int i=0;i<n;i++){
                if(a[cnt]>=2*a[i]){
                    cnt++;
                    if(cnt==n/2)
                        break;
                }
            }
    
            printf("%d
    ",n-cnt);
        }
    }
    
  • 相关阅读:
    软件仓库配置及编译http2.4及文件系统创建实例
    查找、打包、sed实例
    文件管理工具和基础脚本实例
    文件管理实例
    Linux系统中vim设置tab缩进为4个字符
    linux文件管理类命令及实例讲解
    文件元数据信息介绍及修改文件时间
    Linux发行版的系统目录名称命名规则以及用途
    回调和递归
    关于for循环的小案例
  • 原文地址:https://www.cnblogs.com/Yinku/p/10798879.html
Copyright © 2020-2023  润新知