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\n", i, j);
                //printf("%d %d %d %d\n", doll[i].width, doll[i].height, chain[i].width, chain[i].height);
                break;
            }
        if (!fitted)
        {
            //printf("%d %d\n", 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\n", chain_cnt);
    }
    return 0;
}
View Code

 

posted @ 2013-07-03 15:49  金海峰  阅读(447)  评论(0编辑  收藏  举报