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

全部博文(117)

文章存档

2012年(5)

2011年(5)

2010年(46)

2009年(61)

我的朋友

分类:

2009-08-29 16:34:14

 

解题思路

题意:

中文的不用翻译了

思路:

    一开始想用DFS,后来搞半天没搞出来,放弃。后来改用dijkstra。具体思路是:

    先把每个物品放在一个数组goods里,标记每个物品的pricerank 顺便把每个rank存在另一个数组ranklist里。并把ranklist排序,从大到小或者从小到大无所谓。增加一个物品节点,rank跟酋长的一样,使他到每个节点都有一条路,路的权是那些节点本身的price。其他节点间如果可以交换就也建条路。如此,图就建成了。然后枚举每个不相同的rank,作为最高等级,每次用dijkstra前,把那些等级比他高的还有跟它差距大于M的节点通通去掉。然后就是经典的dijkstra,从goods[gonum+1]走到goods[1] ,即从自己加的那个节点到酋长节点。不用任何其他修改。返回sum,就是那条最短路径。 最后找出最短路径中的最小值。

 

源程序
#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include <conio.h>
#define N 110
#define inf 1000000000

struct node
{
    int price;
    int rank;
}goods[N];
int g[N][N];

int dijkstra(int v, int n, int limit, int max)
{
    int i, j, min, minid, k, sum=inf;
    int dis[N], mark[N];

    for(i=1; i<=n; i++)
    {
        mark[i] = 0;
        dis[i] = g[v][i];

        //去掉那些不能用的点
        if((goods[i].rank > max) || (max-goods[i].rank > limit)
            mark[i] = 1;
    }

    mark[v] = 1;
    
    for(i=1; i<n; i++)
    {
        min = inf;
        minid = 0;

        for(j=1; j<=n; j++)
        {
            if(!mark[j] && dis[j] < min)
            {
                min = dis[j];
                minid = j;
            }
        }
        if(!minid)
            break;
        mark[minid] = 1;

        if(minid == 1)
        {
            sum = dis[1];
            break;
        }
        for(j=1; j<=n; j++)
        {
            if(!mark[j] && (min+g[minid][j] < dis[j]))
            {
                dis[j] = min + g[minid][j];
            }
        }
    }
    return sum;
}

int cmp(const void *p, const void *q)
{
    return *(int *)p > *(int *)q ? -1 : 1;
}

int main()
{
    int i, j, k;
    int gonum, n, x, limit, r, result, ranklist[N];

    freopen("in.txt", "r", stdin);
    scanf("%d%d", &limit, &gonum);

    for(i=1; i<=gonum+1; i++)
        for(j=1; j<=gonum+1; j++)
            g[i][j] = inf;
    for(i=1; i<=gonum; i++)        

    {
        scanf("%d%d%d", &goods[i].price, &goods[i].rank, &n);
        ranklist[i] = goods[i].rank;
        for(j=1; j<=n; j++)
        {
            scanf("%d", &x);
            scanf("%d", &g[x][i]);
        }
    }
    qsort(ranklist, gonum+1, sizeof(ranklist[1]), cmp);
    for(i=1; i<=gonum; i++)
        g[gonum+1][i] = goods[i].price;
    goods[gonum+1].rank = goods[1].rank;
    result = goods[1].price;

//枚举每个不相同rank,作为最高等级
    for(i=0; i<gonum+1; i++)
    {
        r = dijkstra(gonum+1, gonum+1, limit, ranklist[i]);
        if(r < result)
            result = r;
        while(ranklist[i+1] == ranklist[i] && i<gonum+1)
            i++;
    }
    printf("%d\n", result);
    getch();
    return 0;
}

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