傳送門:http://poj.org/problem?id=1062
昂貴的聘禮
Time Limit: 1000MS | Memory Limit: 10000K | |
Total Submissions: 53123 | Accepted: 16003 |
Description
年輕的探險家來到了一個印第安部落裏。在那裏他和酋長的女兒相愛了,於是便向酋長去求親。酋長要他用10000個金幣作爲聘禮才答應把女兒嫁給他。探險家拿不出這麼多金幣,便請求酋長降低要求。酋長說:"嗯,如果你能夠替我弄到大祭司的皮襖,我可以只要8000金幣。如果你能夠弄來他的水晶球,那麼只要5000金幣就行了。"探險家就跑到大祭司那裏,向他要求皮襖或水晶球,大祭司要他用金幣來換,或者替他弄來其他的東西,他可以降低價格。探險家於是又跑到其他地方,其他人也提出了類似的要求,或者直接用金幣換,或者找到其他東西就可以降低價格。不過探險家沒必要用多樣東西去換一樣東西,因爲不會得到更低的價格。探險家現在很需要你的幫忙,讓他用最少的金幣娶到自己的心上人。另外他要告訴你的是,在這個部落裏,等級觀念十分森嚴。地位差距超過一定限制的兩個人之間不會進行任何形式的直接接觸,包括交易。他是一個外來人,所以可以不受這些限制。但是如果他和某個地位較低的人進行了交易,地位較高的的人不會再和他交易,他們認爲這樣等於是間接接觸,反過來也一樣。因此你需要在考慮所有的情況以後給他提供一個最好的方案。
爲了方便起見,我們把所有的物品從1開始進行編號,酋長的允諾也看作一個物品,並且編號總是1。每個物品都有對應的價格P,主人的地位等級L,以及一系列的替代品Ti和該替代品所對應的"優惠"Vi。如果兩人地位等級差距超過了M,就不能"間接交易"。你必須根據這些數據來計算出探險家最少需要多少金幣才能娶到酋長的女兒。
爲了方便起見,我們把所有的物品從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
題意:這個題就是找一個符合題意的最短路,設出發點爲S,點1爲目的地,現在S到1有一條路,但是這條路可能很長,所以有一些小路的也是能到達點1的,每個地方都有一個等級,你所走過的路的最大等級差不能超過m。
思路:先說建圖,由樣例下手:S-->1長度10000,還有兩條小路分別是,2-->1長8000,3-->1長5000;S-->2長度1000,還有1條小路是,4-->2長200;相信大家看到這裏就明白圖是怎麼建的了,小路也就相當於兌換物品的捷徑。
因爲題中還有一個等級制度,你所走的路中任意兩點的等級差不能超過m,所以用搜索,記錄每個狀態的最大等級,和最小等級。具體內容看代碼解釋。
PS:在11個月前,剛學最短路,做到這個題的時候,感覺太難放棄了,現在回過頭來補題的時候,看了不一會兒就知道該怎麼寫了,做題能力有了很大提高,繼續加油!。
AC代碼:
#include<stdio.h>
#include<string.h>
#include<string>
#include<iostream>
#include<algorithm>
#include<cmath>
#include<map>
#include<queue>
#include<vector>
#define delta 0.98
#define LL long long
#define inf 0x3f3f3f3f
#define mem(a,b) memset(a,b,sizeof(a))
#define ABS(a) a>=0?a:-a
const double eps=1e-7;
const int N=205;
using namespace std;
struct node1
{
int v,nex,w;
}e[10500];//代碼習慣用數組模擬鏈表 建圖
struct node
{
int s,l,r,num;//s表示當前所在的點,l,r表示這條路上的等級的最小最大值,num表示路長。
};
int head[N],d[N];//d數組存等級
int cnt,S,n,m;//S表示出發點0
void add(int u,int v,int w)
{
e[cnt].v=v;
e[cnt].w=w;
e[cnt].nex=head[u];
head[u]=cnt++;
}
void init()
{
mem(head,-1);
S=cnt=0;
}
int BFS()
{
int ans=inf;//存最小答案
queue<node>q;
node q1,q2;
q1.l=q1.r=d[1];//,因爲終點是1,所以初始等級可以直接設成d[1];
q1.num=q1.s=0;
q.push(q1);
while(!q.empty())
{
q1=q.front();
q.pop();
int u=q1.s;
if(u==1)//如果找到終點
{
if(ans>q1.num) ans=q1.num;//跟新答案
continue;
}
for(int i=head[u];~i;i=e[i].nex)
{
q2=q1;
int v=e[i].v;
if(d[v]<q2.l) q2.l=d[v];//跟新最大最小等級
if(d[v]>q2.r) q2.r=d[v];
if(q2.l+m<q2.r) continue;//不符合的剪枝。
q2.num+=e[i].w;
if(q2.num>=ans) continue;//路長超過答案的,剪枝
q2.s=v;
q.push(q2);
}
}
return ans;
}
int main()
{
while(~scanf("%d%d",&m,&n))
{
init();
int x,a,b;
for(int i=1;i<=n;i++)
{
scanf("%d%d%d",&a,&d[i],&x);
add(S,i,a);//建大路
while(x--)//建小路
{
scanf("%d%d",&a,&b);
add(a,i,b);
}
}
printf("%d\n",BFS());
}
}