Time Limit: 1000MS | Memory Limit: 10000K | |
Total Submissions: 40715 | Accepted: 11839 |
Description
Input
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.把优惠的部分当作路径然后dijkstra,最后比较的是每个点的价值+那个点的最短路径长度。
2.记录进入set时的每个点的前面节点,为了比较level时使用,这个节点以及这个节点的前面节点都比较level之后通过才行。
代码:
#include <iostream> #include <algorithm> #include <cmath> #include <vector> #include <string> #include <cstring> #pragma warning(disable:4996) using namespace std; int MAX; int edge[1005][1005]; int vist[1005],level[1005],minidis[1005],value[1005],pre[1005]; int level_diff,good_num; void init() { memset(edge,-1,sizeof(edge)); } void init2() { int i; for(i=1;i<=good_num;i++) { minidis[i]=MAX; pre[i]=1; vist[i]=0; } } void dijkstra(int i) { int j,k; int position=i; vist[position]=1; minidis[position]=0; for(j=1;j<=good_num-1;j++)//一共要添加进num-1个点 { for(k=1;k<=good_num;k++) { if(vist[k]==0 && abs(level[i]-level[k])<=level_diff && edge[position][k]!=-1 && minidis[position]+edge[position][k] < minidis[k])//新填入的点更新minidis { int flag=0,temp=position; while(temp!=1) { if(abs(level[temp]-level[k])>level_diff) { flag=1; break; } else { temp=pre[temp]; } } if(flag==0) { minidis[k]=minidis[position]+edge[position][k]; pre[k]=position; } } } int min_value=MAX,min_pos=0; for(k=1;k<=good_num;k++) { if(vist[k]==0 && minidis[k]<min_value)//比较出最小的那一个作为新添入的店 { min_value = minidis[k]; min_pos = k; } } vist[min_pos]=1; position=min_pos; } } int main() { int i,j,replace; scanf("%d%d",&level_diff,&good_num); init(); for(i=1;i<=good_num;i++) { scanf("%d%d%d",&value[i],&level[i],&replace); int temp; for(j=1;j<=replace;j++) { scanf("%d",&temp); scanf("%d",&edge[i][temp]); } } MAX=value[1]; init2(); dijkstra(1); int ans=MAX; for(i=1;i<=good_num;i++) { ans=min(ans,minidis[i]+value[i]); } cout<<ans<<endl; return 0; }
版权声明:本文为博主原创文章,未经博主允许不得转载。
原文:http://blog.csdn.net/u010885899/article/details/47283491