昂贵的聘礼 POJ162

                                                              昂贵的聘礼      


题目链接:Click Here~

题目描述:

   不过探险家没必要用多样东西去换一样东西,因为不会得到更低的价格。探险家现在很需要你的帮忙,让他用最少的金币娶到自己的心上人。另外他要告诉你的是,在这个部落里,等级观念十分森严。地位差距超过一定限制的两个人之间不会进行任何形式的直接接触,包括交易。他是一个外来人,所以可以不受这些限制。但是如果他和某个地位较低的人进行了交易,地位较高的的人不会再和他交易,他们认为这样等于是间接接触,反过来也一样。因此你需要在考虑所有的情况以后给他提供一个最好的方案。
为了方便起见,我们把所有的物品从1开始进行编号,酋长的允诺也看作一个物品,并且编号总是1。每个物品都有对应的价格P,主人的地位等级L,以及一系列的替代品Ti和该替代品所对应的"优惠"Vi。如果两人地位等级差距超过了M,就不能"间接交易"。你必须根据这些数据来计算出探险家最少需要多少金币才能娶到酋长的女儿。


题目分析:

   可以联系到最短路,但一开始做的时候考虑错了。虽然,知道区间是[m-rank,m+rank](rank为酋长的等级),但开始我并没有分区间讨论,而是咔咔的写了一个混乱的代码,而且是一个下午的时间。过后wrong后看讨论区才知道是要分区间讨论的。而知道了这两点后就不难了。


思路分析:

    最短路 + 区间讨论

    但是要注意此题特坑人的是建图的时候,因为只有在知道了可以得到要替代的物品后,才能得到优惠的价格。所以,建图是从要替代品到要购买的物品。


 
#include <iostream>
#include <algorithm>
#include <cstdio>
#include <cstring>
using namespace std;

const int N = 100 + 5;
const int INF = 9999999;
int n,rank[N],graph[N][N];
int Dijkstra(int s,int e)
{
    int dist[N];
    bool vst[N];
    memset(vst,0,sizeof(vst));
    for(int i = 1;i <= n;++i){
      if(rank[i]>=s&&rank[i]<=e)
        dist[i] = graph[0][i];
      else
        dist[i] = INF;
    }
    for(int i = 0;i < n;++i){
        int x,m = INF;
        for(int y = 1;y <= n;++y)
          if((rank[y]>=s&&rank[y]<=e)&&!vst[y]&&dist[y] < m)
            m = dist[x=y];
        if(m == INF)
          break;
        vst[x] = 1;
        for(int y = 1;y <= n;++y)
          if((rank[y]>=s&&rank[y]<=e)&&(dist[y] > dist[x]+graph[x][y]))
            dist[y] = dist[x] + graph[x][y];
    }
    return dist[1];
}
int main()
{
//    freopen("Input.txt","r",stdin);
    int m,t,num,p;
    while(scanf("%d%d",&m,&n),(n||m))
    {
        for(int i = 0;i <= n;++i)
          for(int j = 0;j <= n;++j)
            graph[i][j] = INF;
        for(int i = 1;i <= n;++i)
        {
            scanf("%d%d%d",&graph[0][i],&rank[i],&t);
            while(t--)
            {
                scanf("%d%d",&num,&p);
                graph[num][i] = p;
            }
        }
        int tmp,ans = graph[0][1];
        for(int i = rank[1]-m;i <= rank[1];++i)
        {
            tmp = Dijkstra(i,i+m);   //printf("tmp = %d\n",tmp);
            if(tmp < ans)
               ans = tmp;
        }
        printf("%d\n",ans);
    }
    return 0;
}
        

  



  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值