升降梯上
电梯每移动一层,需要花费2秒钟时间,而手柄从一个控制槽扳到相邻的槽,需要花费1秒钟时间。探险队员现在在1层,并且想尽快到达N层,他们想知道从1层到N层至少需要多长时间?
思路:建边,将时间当作边权用spfa求最短路,将二维的点弄成一维
(可以用dp,但因为可以下降,所以有后效性,不能包括所有的状态,必须循环跑dp直到答案不变)
#include<bits/stdc++.h> using namespace std; const int maxn=20000+10,maxm=5e5+10,inf=0x3f3f3f3f; int a[maxn]; struct Edge{ int to,next,w; }e[maxm]; int head[maxm],tot=0; void Insert(int a,int b,int c){ e[++tot].to=b; e[tot].next=head[a]; e[tot].w=c; head[a]=tot; } bool inq[maxn]; int d[maxn]; void spfa(int s){ memset(d,0x3f,sizeof(d)); d[s]=0; queue<int> q; q.push(s); inq[s]=1; while(!q.empty()){ int u=q.front(); q.pop(); inq[u]=0; for(int i=head[u];i;i=e[i].next){ int v=e[i].to; if(d[v]>d[u]+e[i].w){ d[v]=d[u]+e[i].w; if(!inq[v]){ q.push(v); inq[v]=1; } } } } } int main(){ int n,m; scanf("%d%d",&n,&m); int s; for(int i=1;i<=m;i++) { scanf("%d",&a[i]); if(a[i]==0) s=i; } for(int i=1;i<=n;i++){ for(int j=1;j<m;j++){ int x=(i-1)*m+j; Insert(x,x+1,1); Insert(x+1,x,1); } } for(int i=1;i<=n;i++){ for(int j=1;j<=m;j++){ int v=i+a[j]; if(v<1||v>n) continue; int x=(i-1)*m+j; int y=(v-1)*m+j; Insert(x,y,2*abs(a[j])); } } spfa(s);//因为起点一定是第一行第j列,(1-1)×m+s==s int ans=inf; for(int i=1;i<=m;i++){ int x=(n-1)*m+i; ans=min(ans,d[x]); }//枚举最后一行的扳手位置 if(ans!=inf) printf("%d\n",ans); else printf("-1\n"); return 0; }
原文:https://www.cnblogs.com/HZOIDJ123/p/13251969.html