Chinaunix首页 | 论坛 | 博客
  • 博客访问: 477247
  • 博文数量: 117
  • 博客积分: 3195
  • 博客等级: 中校
  • 技术积分: 1156
  • 用 户 组: 普通用户
  • 注册时间: 2009-08-04 01:44
文章分类

全部博文(117)

文章存档

2012年(5)

2011年(5)

2010年(46)

2009年(61)

我的朋友

分类:

2009-09-30 20:10:48

解题思路

题意:

       在一个格子地图上有n个小人(m)n个房子(H),每个人每次能向水平方向或者竖直方向移动一个格子,每移动一格要付1美元,直到他进入到房子里,每个房子只能有一个小人。

       要使这n个小人进入n个不同的房子里,我们要计算出所要付的最小费用。

 

思路:

       因为每个人对应于一个房子,所以想到二分匹配。这是个求二分图的最小权匹配问题。

建图的时候,先记录每个mH的位置,再两两算出每个mH的曼哈顿距离作为权。

先分别用一个很大的数inf去减每条边的权,转化为最大权匹配问题。然后用KM算法求解。最后将求得的最大权用n*inf减去就可以了。

至于KM算法怎么求。这就比较麻烦了,好几天才把他搞懂。有空再写。

源程序

 

#include <stdio.h>
#include <string.h>
#include <math.h>
#include <conio.h>
#define N 200
#define inf 10000000

typedef struct
{
    int x, y;
}point;
point house[N], men[N];
int g[N][N], lx[N], ly[N], usedx[N], usedy[N], mat[N];
/* g: 图 ,lx: 左边各点顶标,ly: 右边各点顶标,
    usedx:给左边各点做标记,usedy:给右边各点做标记, mat:和右边点相匹配的点
*/


//找增广路径,条件是未访问过,而且两点顶标之和要等于两点之间的权

int find(int k, int n)
{
    int i;

    usedx[k] = 1;
    for(i=1; i<=n; i++)
    {
        if(!usedy[i] && lx[k] + ly[i] == g[k][i])
        {
            usedy[i] = 1;
            if(mat[i] == -1 || find(mat[i], n))
            {
                mat[i] = k;
                return 1;
            }
        }
    }
    return 0;
}

void km(int n)
{
    int i, j, d, sum;

    memset(lx, 0, sizeof(lx));
    memset(ly, 0, sizeof(ly));
    memset(mat, -1, sizeof(mat));

    //初始化可行顶标
    //令lx[i]为所有与顶点i关联的边的最大权
    for(i=1; i<=n; i++)
    {
        lx[i] = g[i][1];
        for(j=2; j<=n; j++)
        {
            lx[i] = lx[i] > g[i][j] ? lx[i] : g[i][j];
        }
    }

    for(int k=1; k<=n; k++)
    {
        memset(usedx, 0, sizeof(usedx));
        memset(usedy, 0, sizeof(usedy)); //善前

        while(!find(k, n)) //未找到完备匹配
        {
            for(i=1, d = inf*5; i<=n; i++)
                if(usedx[i]) //左边点在交错树中
                    for(j=1; j<=n; j++)
                        if(!usedy[j]) //右边点不在交错树中
                            d = d < lx[i] + ly[j] - g[i][j] ? d : lx[i] + ly[j] - g[i][j];


            for(i=1; i<=n; i++) //修改左边各点顶标
                if(usedx[i])
                    lx[i] -= d;
            for(i=1; i<=n; i++) //修改右边各点顶标
                if(usedy[i])
                    ly[i] += d;
            memset(usedx, 0, sizeof(usedx)); //为下次寻找善前
            memset(usedy, 0, sizeof(usedy));
        }
    }
    sum = 0;
    for(i=1; i<=n; i++)
        sum += lx[i] + ly[i];
        //sum += abs(house[i].y - men[mat[i]].y) + abs(house[i].x - men[mat[i]].x);

    sum = n * inf - sum;
    printf("%d\n", sum);
}

int main()
{
    int i, j;
    int n, m, k1, k2;
    char ch;

    freopen("in.txt", "r", stdin);
    while(scanf("%d%d", &n, &m) != EOF)
    {
        if(!n && !m)
            break;
        memset(house, 0, sizeof(house));
        memset(men, 0, sizeof(men));
        
        k1 = 0;
        k2 = 0;
        for(i=1; i<=n; i++)
        {
            getchar();
            for(j=1; j<=m; j++)
            {
                ch = getchar();

                //记录每个H跟m的位置
                if(ch == 'H')
                {
                    house[++k1].x = j;
                    house[k1].y = i;
                }
                if(ch == 'm')
                {
                    men[++k2].x = j;
                    men[k2].y = i;
                }
            }
        }

        memset(g, 0, sizeof(g));
        for(i=1; i<=k1; i++)
        {
            for(j=1; j<=k2; j++)
            {
                //用一个很大的数去减每个H跟m的曼哈顿距离,把问题转化为最大权问题
                g[i][j] = inf-(abs(house[i].y - men[j].y) + abs(house[i].x - men[j].x));
            }
        }
        km(k1);
    }
    
    getch();
    return 0;
}


阅读(1458) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~