zoukankan      html  css  js  c++  java
  • poj3636

    题意:每个物品有两个属性:长和宽(长宽不可互换)。如果一个物品的长和宽均大于另一个物品,则这个物品可以罩住另一个物品,用这种罩住物品的方法将物品分组,一组之内的物品可以一个罩住一个的全部罩起来。问最少分成几组?

    分析:通常这种问题是问物品最多的一组有多少个,这个问题则稍有不同。有人说要用到Dilworth定理,个人认为没什么关系。我们的做法是将物品先按照长从小到大排序。还有一个数组,用来存储当前每个分组的罩在最外层的长和宽,开始该数组为空。然后从小到大每次取出所有长度相等的物品,扣在当前的分组数组中的物品上,更新数组中存储的长宽值。这样长度相等的一起扣,在判断能不能扣的上的时候就不需要考虑长度了,之前扣好的都是长度小于当前物品的。在扣的过程中要注意发挥每个物品的最大能力,即让每个物品扣住刚好比它小一点点的物品。因为假设如果物品X不去扣比它小一点的那个物品A,而去扣比它小很多的那个物品B。而另一个物品Y,本来只能扣B不能扣A,又因为B已经被X扣了,Y只能单成一组,造成组数增加。在实际操作中是先拿当前长度的那个宽度最大的,发挥其最大作用,然后宽度次大的……为了程序实现起来简便,我们可以直接将所有物品按长度升序,宽度降序来排序,这样每次取出一个物品加入到分组中,就相当与我们刚才的操作了。

    #include <cstdio>
    #include <algorithm>
    using namespace std;
    
    #define MAX_DOLL_NUM 20004
    
    struct Doll
    {
        int width, height;
    }doll[MAX_DOLL_NUM], chain[MAX_DOLL_NUM];
    
    int doll_num;
    int chain_cnt;
    
    bool operator < (const Doll &a, const Doll &b)
    {
        if (a.width == b.width)
            return a.height > b.height;
        return a.width < b.width;
    }
    
    void input()
    {
        scanf("%d", &doll_num);
        for (int i = 0; i < doll_num; i++)
            scanf("%d%d", &doll[i].width, &doll[i].height);
    }
    
    bool fit(Doll a, Doll b)
    {
        return a.width > b.width && a.height > b.height;
    }
    
    void work()
    {
        chain_cnt = 0;
        for (int i = 0; i < doll_num; i++)
        {
            bool fitted = false;
            for (int j = 0; j < chain_cnt; j++)
                if (fit(doll[i], chain[j]))
                {
                    fitted = true;
                    chain[j] = doll[i];
                    //printf("%d %d
    ", i, j);
                    //printf("%d %d %d %d
    ", doll[i].width, doll[i].height, chain[i].width, chain[i].height);
                    break;
                }
            if (!fitted)
            {
                //printf("%d %d
    ", doll[i].width, doll[i].height);
                chain[chain_cnt++] = doll[i];
            }
        }
    }
    
    int main()
    {
        int t;
        scanf("%d", &t);
        while (t--)
        {
            input();
            sort(doll, doll + doll_num);
            work();
            printf("%d
    ", chain_cnt);
        }
        return 0;
    }
    View Code
  • 相关阅读:
    php设计模式之观察者模式
    git tag 相关命令
    git 命令
    phpstudy 配置本地站点的ssl证书
    b
    __invoke,try{}catch(){},microtime(),is_callable()
    json_encode 中文不乱码
    php ::class
    yii 2 美化url
    JNIjw03
  • 原文地址:https://www.cnblogs.com/rainydays/p/3169645.html
Copyright © 2011-2022 走看看