昂貴的聘禮
Description
年輕的探險家來到了一個印第安部落裏。在那裏他和酋長的女兒相愛了,於是便向酋長去求親。酋長要他用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 Source |
[Submit] [Go Back] [Status] [Discuss]
題解: 很簡單的一道圖論題,其實和動態規劃的思路差不多,只要把樣例的關係圖畫出來,然後就可以看出解題思路是最短路,題目的範圍很小,可以很清楚使用最短路求解。#include <queue>
#include <cstdio>
#include <iostream>
#include <algorithm>
using namespace std;
typedef pair<int, int> PII;
const int maxn = 110;
const int INF = 0x3f3f3f3f;
int Edge[maxn][maxn], n, m;
struct People{
int p, l, x;
}Peo[maxn];
void init(){
for(int i = 0; i <= n; i++)
for(int j = 0; j <= n; j++){
Edge[i][j] = Edge[j][i] = INF;
}
for(int i = 1; i <= n; i++){
scanf("%d%d%d", &Peo[i].p, &Peo[i].l, &Peo[i].x);
for(int j = 0; j < Peo[i].x; j++){
int t, v;
scanf("%d%d", &t, &v);
Edge[t][i] = v;
}
Edge[0][i] = Peo[i].p;
}
}
int d[maxn], vis[maxn];
int dijkstra(){
priority_queue<PII, vector<PII>, greater<PII> > que;
fill(d, d + n + 1, INF);
d[0] = 0;
que.push(PII(0, 0));
while(!que.empty()){
PII p = que.top();
que.pop();
int v = p.second;
if(d[v] < p.first) continue;
for(int k = 0; k <= n; k++){
if(k == v || vis[k]) continue;
if(d[k] > d[v] + Edge[v][k]){
d[k] = d[v] + Edge[v][k];
que.push(PII(d[k], k));
}
}
}
return d[1];
}
int main(){
while(~scanf("%d%d", &m, &n)){
init();
int ans = INF;
for(int i = 1; i <= n; i++){
int minlevel = Peo[i].l;
for(int j = 1; j <= n; j++){
vis[j] = 0;
if(Peo[j].l - minlevel > m || Peo[j].l < minlevel) vis[j] = 1;
}
int res = dijkstra();
ans = min(ans, res);
}
printf("%d\n", ans);
}
return 0;
}