zoukankan      html  css  js  c++  java
  • 昂贵的聘礼 POJ

    题目:

    年轻的探险家来到了一个印第安部落里。在那里他和酋长的女儿相爱了,于是便向酋长去求亲。酋长要他用10000个金币作为聘礼才答应把女儿嫁给他。探险家拿不出这么多金币,便请求酋长降低要求。酋长说:"嗯,如果你能够替我弄到大祭司的皮袄,我可以只要8000金币。如果你能够弄来他的水晶球,那么只要5000金币就行了。"探险家就跑到大祭司那里,向他要求皮袄或水晶球,大祭司要他用金币来换,或者替他弄来其他的东西,他可以降低价格。探险家于是又跑到其他地方,其他人也提出了类似的要求,或者直接用金币换,或者找到其他东西就可以降低价格。不过探险家没必要用多样东西去换一样东西,因为不会得到更低的价格。探险家现在很需要你的帮忙,让他用最少的金币娶到自己的心上人。另外他要告诉你的是,在这个部落里,等级观念十分森严。地位差距超过一定限制的两个人之间不会进行任何形式的直接接触,包括交易。他是一个外来人,所以可以不受这些限制。但是如果他和某个地位较低的人进行了交易,地位较高的的人不会再和他交易,他们认为这样等于是间接接触,反过来也一样。因此你需要在考虑所有的情况以后给他提供一个最好的方案。 
    为了方便起见,我们把所有的物品从1开始进行编号,酋长的允诺也看作一个物品,并且编号总是1。每个物品都有对应的价格P,主人的地位等级L,以及一系列的替代品Ti和该替代品所对应的"优惠"Vi。如果两人地位等级差距超过了M,就不能"间接交易"。你必须根据这些数据来计算出探险家最少需要多少金币才能娶到酋长的女儿。 

    Input

    输入第一行是两个整数M,N(1 <= N <= 100),依次表示地位等级差距限制和物品的总数。接下来按照编号从小到大依次给出了N个物品的描述。每个物品的描述开头是三个非负整数P、L、X(X < N),依次表示该物品的价格、主人的地位等级和替代品总数。接下来X行每行包括两个整数T和V,分别表示替代品的编号和"优惠价格"。

    Output

    输出最少需要的金币数。

    Sample Input

    1 4
    10000 3 2
    2 8000
    3 5000
    1000 2 1
    4 200
    3000 2 1
    4 200
    50 2 0
    

    Sample Output

    5250

    分析:
    由于最终要换到1号物品,第一反应是逆向求1号物品换到其它物品的花费,再转换成换到1号物品的花费,但是由于交换物品时的等级限制,这样处理会很麻烦
    于是想到,假设一个0号物品,求由0号物品换到1号物品的花费
    关于等级限制,无法确定酋长是否是最高等级,于是可以枚举最低等级(每个最短路一定有一个最低等级),进行n次最短路操作,则可以得到所有交换情况。注意:每次选取dis最小的点时,要判断等级是不是在范围之内,同时松弛的时候也是也判断等级范围。
    代码:
     1 #include <iostream>
     2 #include <algorithm>
     3 #include <string>
     4 using namespace std;
     5 const int inf = 0x3f3f3f3f;
     6 struct obj
     7 {
     8     int v;
     9     int d;
    10 }a[110];
    11 int m, n;
    12 int g[110][110];
    13 int dis[110];
    14 bool vis[110];
    15 
    16 void dj(int low)
    17 {
    18     for (int i = 1; i <= n; i++)
    19         dis[i] = g[0][i];
    20     memset(vis, 0, sizeof(vis));
    21 
    22     for (int i = 1; i <= n; i++)
    23     {
    24         int u = -1, minn = inf;
    25         for (int j = 1; j <= n; j++)
    26             if (!vis[j] && dis[j] < minn)
    27                 minn = dis[j], u = j;
    28         if (u == -1) continue;
    29         vis[u] = 1;
    30         if (a[u].d < low || a[u].d - low > m) continue;
    31         for (int j = 1; j <= n; j++)
    32             if (!vis[j] && dis[j] > dis[u] + g[u][j] && a[j].d - low >= 0 && a[j].d - low <= m)
    33                 dis[j] = dis[u] + g[u][j];
    34     }
    35 }
    36 
    37 int main()
    38 {
    39     cin >> m >> n;
    40     int num;
    41 
    42     memset(g, inf, sizeof(g));
    43     for (int i = 1; i <= n; i++)
    44     {
    45         cin >> a[i].v >> a[i].d >> num;
    46         g[0][i] = a[i].v;
    47         int t, v;
    48         for (int j = 1; j <= num; j++)
    49         {
    50             cin >> t >> v;
    51             g[t][i] = v;
    52         }
    53     }
    54     int ans = a[1].v;
    55 
    56     for (int i = 1; i <= n; i++)
    57     {
    58         dj(a[i].d);
    59         ans = min(ans, dis[1]);
    60     }
    61     cout << ans << endl;
    62 }


  • 相关阅读:
    Kaldi的data目录解析
    Kaldi的nnet3
    Kaldi中的Chain模型
    Karel版本的nnet1
    Dan版本的nnet2
    MFCC/Filter Bank的提取流程
    【算法专题】工欲善其事必先利其器—— C++ STL中vector(向量/不定长数组)的常用方法总结
    App 设计技巧
    js判断是否在微信浏览器中打开
    WebApi 跨域
  • 原文地址:https://www.cnblogs.com/liuwenhan/p/11421244.html
Copyright © 2011-2022 走看看