昂貴的聘禮
time limit:1000ms
memory limit:10000k
total submissions:29628
accepted:8296
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 4sample output10000 3 2
2 8000
3 5000
1000 2 1
4 200
3000 2 1
4 200
50 2 0
5250source
浙江如果你還不知道什麼叫最短路請先看:
演算法:dijkstra + 列舉
難點:等級的處理 靠列舉
思路來自:
大牛的思路:對於從u
點出發到
w點的路徑中,他會跟很多等級的人交易,然而必須滿足在路徑中的點等級差不很超過乙個
m值,那麼怎麼對這樣的問題求解呢?我沒看報告前是很疑惑的!
假設如果給這條路徑加上乙個附加條件的話,情況可能就有所變化了,要求最短路中的所有點的等級在乙個區間內[a,b]
,如果能夠很好的給出這個區間的話,只要對圖中的點進行上篩選即可了。
這個區間的確定顯然不是隨便的,那麼就要根據一定的條件了,從題意中我們知道,最後所有的最短路都會匯集在1
號點,也就是說
1號點是所有最短路都存在的點,好了,這個條件很重要,這樣我們就可以依照
1號點來給定區間了,比如
1號點等級為
lev,那麼也就是說在所有最短路的這些點都必須滿足在
[lev-m,lev+m]
這個區間裡面。好了,可能你會迫不及待將這個區間作為最後的區間,在想想,如果在這個區間內出現的兩個點的他們之間的等級差超過了
m值(這是存在的),顯然,不符合題意了,所以這個區間還有繼續縮小。其實只要稍微動動腦子,就可以找出這樣的區間
[lev-m,lev],[lev-m+1,lev+1],... ...,[lev,lev+m]
,首先這些區間都滿足大區間的條件,而且如果將這些區間的某個作為篩選條件的話,在這個區間內的任意兩個點的等級都不會超過
m值,這就是很特別的地方了,我也是在這裡卡了的。
好了,講完了,只需列舉區間,然後篩選點,求最短路就行了。
大牛的**:
1 //最短路徑
——dijkstra
演算法
2 //此題的關鍵在於等級限制的處理,最好的辦法是採用列舉,即假設酋長等級為
5,等級限制為
2,那麼需要列舉等級從
3~5,4~6,5~7
3 //從滿足改等級範圍的結點組成的子圖中用
dijkstra
來算出最短路徑
4 //小結,通過列舉的方式可以消除一些圖與圖之間的限制
好吧。。。。照著大牛的**寫的了
#include#include#includeusing namespace std;const int maxn = 110;
const int inf = 200000000;
int w[maxn][maxn]; //w[i][j]表示物品i換j的花費
int dist[maxn]; //dist[i]表示起點到i的花費
bool vis[maxn];
int level[maxn], value[maxn]; //等級與價值
bool can_change[maxn]; //滿足等級限制的標記陣列
int limit_level, n; //limit_level 表示限制的等級
int dijkstra()
vis[x] = true; //標記點x
for(int y=1;y<=n;y++) // 對於從x出發的所有邊 (x, y)更新dist
}for(int y=1;y<=n;y++)
return mini_cost; //返回最小值
}int main() }
for(int i=1;i<=n;i++) }
int king_level = level[1]; //酋長的等級
int m, minicost = inf;
for(int i=0;i<=limit_level;i++) //列舉
minicost = min(minicost, dijkstra());
} printf("%d\n",minicost);
return 0;
}
poj 1062 昂貴的聘禮
這個題是乙個簡單的樹形dp,只是有乙個地位差限制,可以列舉可行區間,在可行區間裡進行dp.include include include include include include include include include include includeusing namespace st...
poj1062 昂貴的聘禮
做這題代價確實很昂貴,從昨天晚上一直到今天上午都在debug,今天上午提交還出現各種wa。比較明顯的最短路,有幾個地方要注意一下,首先建圖有個地方要注意以下,這題是用臨界矩陣儲存的有向圖,而且是由交換品指向購買品,然後將虛節點設為購買品的價值,最後列舉每乙個可能地位區間di求最短路,因為對於區間 c...
昂貴的聘禮(POJ1062)
這個玩了好久啊,是我第一次寫出這樣的最短路徑。還是參考了其它人的才出來,注意的有 1 是單向圖,2 把不滿足權值的提前全初始化。其實不用topo排序也可的。關鍵 void dfs int k include include include include include include includ...